/* -*-mode: C; fill-column: 78; c-basic-offset: 4; compile-command: "make MEMDEBUG=TRUE"; -*- */ /* * Copyright 2000-2008 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 #include #include #include #include /* gethostbyname */ #include //#include #include #include #include #include #include #include #include #include #include "linuxmain.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 "server.h" #include "memstream.h" #include "util.h" #include "dbgutil.h" #ifdef CURSES_SMALL_SCREEN # define MENU_WINDOW_HEIGHT 1 # define BOARD_OFFSET 0 #else # define MENU_WINDOW_HEIGHT 5 /* three lines plus borders */ # define BOARD_OFFSET 1 #endif #ifndef CURSES_CELL_HT # define CURSES_CELL_HT 1 #endif #ifndef CURSES_CELL_WIDTH # define CURSES_CELL_WIDTH 2 #endif #ifndef CURSES_MAX_HEIGHT # define CURSES_MAX_HEIGHT 40 #endif #ifndef CURSES_MAX_WIDTH //# define CURSES_MAX_WIDTH 50 # define CURSES_MAX_WIDTH 70 #endif #define INFINITE_TIMEOUT -1 #define BOARD_SCORE_PADDING 3 typedef XP_Bool (*CursesMenuHandler)(CursesAppGlobals* globals); typedef struct MenuList { CursesMenuHandler handler; char* desc; char* keyDesc; char key; } MenuList; static XP_Bool handleQuit( CursesAppGlobals* globals ); static XP_Bool handleRight( CursesAppGlobals* globals ); static XP_Bool handleSpace( CursesAppGlobals* globals ); static XP_Bool handleRet( CursesAppGlobals* globals ); static XP_Bool handleHint( CursesAppGlobals* globals ); 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 handleCommit( CursesAppGlobals* globals ); static XP_Bool handleFlip( CursesAppGlobals* globals ); static XP_Bool handleToggleValues( CursesAppGlobals* globals ); static XP_Bool handleBackspace( CursesAppGlobals* globals ); static XP_Bool handleUndo( CursesAppGlobals* globals ); static XP_Bool handleReplace( CursesAppGlobals* globals ); static XP_Bool handleRootKeyShow( CursesAppGlobals* globals ); static XP_Bool handleRootKeyHide( CursesAppGlobals* globals ); static XP_Bool handleJuggle( CursesAppGlobals* globals ); static XP_Bool handleHide( CursesAppGlobals* globals ); static XP_Bool handleAltLeft( CursesAppGlobals* globals ); static XP_Bool handleAltRight( CursesAppGlobals* globals ); static XP_Bool handleAltUp( CursesAppGlobals* globals ); static XP_Bool handleAltDown( CursesAppGlobals* globals ); MenuList g_sharedMenuList[] = { { handleQuit, "Quit", "Q", 'Q' }, { handleRight, "Tab right", "", '\t' }, { handleSpace, "Raise focus", "", ' ' }, { handleRet, "Click/tap", "", '\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", "", 8 }, { handleUndo, "Undo prev", "U", 'U' }, { handleReplace, "uNdo cur", "N", 'N' }, { NULL, NULL, NULL, '\0'} }; MenuList g_boardMenuList[] = { { handleAltLeft, "Force left", "{", '{' }, { handleAltRight, "Force right", "}", '}' }, { handleAltUp, "Force up", "_", '_' }, { handleAltDown, "Force down", "+", '+' }, { NULL, NULL, NULL, '\0'} }; MenuList g_scoreMenuList[] = { #ifdef KEYBOARD_NAV #endif { NULL, NULL, NULL, '\0'} }; MenuList g_trayMenuList[] = { { handleJuggle, "Juggle", "G", 'G' }, { handleHide, "[un]hIde", "I", 'I' }, { handleAltLeft, "Divider left", "{", '{' }, { handleAltRight, "Divider right", "}", '}' }, { NULL, NULL, NULL, '\0'} }; #ifdef CURSES_SMALL_SCREEN MenuList g_rootMenuListShow[] = { { handleRootKeyShow, "Press . for menu", "", '.' }, { NULL, NULL, NULL, '\0'} }; MenuList g_rootMenuListHide[] = { { handleRootKeyHide, "Clear menu", ".", '.' }, { NULL, NULL, NULL, '\0'} }; #endif static CursesAppGlobals g_globals; /* must be global b/c of SIGWINCH_handler */ 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 ); static void countMenuLines( MenuList** menuLists, int maxX, int padding, int* nLinesP, int* nColsP ); static void drawMenuFromList( WINDOW* win, MenuList** menuLists, int nLines, int padding ); static CursesMenuHandler getHandlerForKey( MenuList* list, char ch ); #ifdef MEM_DEBUG # define MEMPOOL params->util->mpool, #else # define MEMPOOL #endif /* extern int errno; */ static void cursesUserError( CursesAppGlobals* globals, char* format, ... ) { char buf[512]; va_list ap; va_start( ap, format ); vsprintf( buf, format, ap ); (void)cursesask( globals, buf, 1, "OK" ); va_end(ap); } /* cursesUserError */ static XP_S16 curses_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* XP_UNUSED(pi), XP_U16 playerNum, const XP_UCHAR4* texts, XP_U16 nTiles ) { CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; char query[128]; XP_S16 index; char* playerName = globals->cGlobals.params->gi.players[playerNum].name; sprintf( query, "Pick tile for %s! (Tab or type letter to select\n" "then hit .)", playerName ); index = curses_askLetter( globals, query, texts, nTiles ); return index; } /* util_userPickTile */ static void curses_util_userError( XW_UtilCtxt* uc, UtilErrID id ) { CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; XP_Bool silent; XP_UCHAR* message = linux_getErrString( id, &silent ); if ( silent ) { XP_LOGF( "silent userError: %s", message ); } else { cursesUserError( globals, message ); } } /* curses_util_userError */ static XP_Bool curses_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id, XWStreamCtxt* stream ) { CursesAppGlobals* globals; char* question; char* answers[3]; short numAnswers = 0; XP_Bool freeMe = XP_FALSE; XP_Bool result; XP_U16 okIndex = 1; switch( id ) { case QUERY_COMMIT_TURN: question = strFromStream( stream ); freeMe = XP_TRUE; answers[numAnswers++] = "Cancel"; answers[numAnswers++] = "Ok"; break; case QUERY_COMMIT_TRADE: question = "Commit trade?"; answers[numAnswers++] = "Cancel"; answers[numAnswers++] = "Ok"; break; case QUERY_ROBOT_MOVE: case QUERY_ROBOT_TRADE: question = strFromStream( stream ); freeMe = XP_TRUE; answers[numAnswers++] = "Ok"; okIndex = 0; break; default: XP_ASSERT( 0 ); return 0; } globals = (CursesAppGlobals*)uc->closure; result = cursesask( globals, question, numAnswers, answers[0], answers[1], answers[2] ) == okIndex; if ( freeMe ) { free( question ); } return result; } /* curses_util_userQuery */ static void curses_util_trayHiddenChange( XW_UtilCtxt* XP_UNUSED(uc), XW_TrayVisState XP_UNUSED(state), XP_U16 XP_UNUSED(nVisibleRows) ) { /* nothing to do if we don't have a scrollbar */ } /* curses_util_trayHiddenChange */ static void cursesShowFinalScores( CursesAppGlobals* globals ) { XWStreamCtxt* stream; XP_UCHAR* text; stream = mem_stream_make( MPPARM(globals->cGlobals.params->util->mpool) globals->cGlobals.params->vtMgr, globals, CHANNEL_NONE, NULL ); server_writeFinalScores( globals->cGlobals.game.server, stream ); text = strFromStream( stream ); (void)cursesask( globals, text, 1, "Ok" ); free( text ); } /* cursesShowFinalScores */ static void curses_util_notifyGameOver( XW_UtilCtxt* uc ) { CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; board_draw( globals->cGlobals.game.board ); /* game belongs in cGlobals... */ if ( globals->cGlobals.params->printHistory ) { catGameHistory( &globals->cGlobals ); } if ( globals->cGlobals.params->quitAfter ) { globals->timeToExit = XP_TRUE; } else if ( globals->cGlobals.params->undoWhenDone ) { server_handleUndo( globals->cGlobals.game.server ); } else { cursesShowFinalScores( globals ); } } /* curses_util_notifyGameOver */ static XP_Bool curses_util_hiliteCell( XW_UtilCtxt* XP_UNUSED(uc), XP_U16 XP_UNUSED(col), XP_U16 XP_UNUSED(row) ) { return XP_TRUE; } /* curses_util_hiliteCell */ static XP_Bool curses_util_engineProgressCallback( XW_UtilCtxt* XP_UNUSED(uc) ) { return XP_TRUE; } /* curses_util_engineProgressCallback */ static void curses_util_setTimer( XW_UtilCtxt* uc, XWTimerReason why, XP_U16 when, XWTimerProc proc, void* closure ) { CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; globals->cGlobals.timerProcs[why] = proc; globals->cGlobals.timerClosures[why] = closure; globals->nextTimer = util_getCurSeconds(uc) + when; } /* curses_util_setTimer */ static void curses_util_requestTime( XW_UtilCtxt* uc ) { /* I've created a pipe whose read-only end is plugged into the array of fds that my event loop polls so that I can write to it to simulate post-event on a more familiar system. It works, so no complaints! */ CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; (void)write( globals->timepipe[1], "!", 1 ); } /* curses_util_requestTime */ static void initCurses( CursesAppGlobals* globals ) { WINDOW* mainWin; WINDOW* menuWin; WINDOW* boardWin; int width, height; /* ncurses man page says most apps want this sequence */ mainWin = initscr(); cbreak(); noecho(); nonl(); intrflush(stdscr, FALSE); keypad(stdscr, TRUE); /* effects wgetch only? */ getmaxyx(mainWin, height, width ); XP_LOGF( "getmaxyx->w:%d; h:%d", width, height ); if ( height > CURSES_MAX_HEIGHT ) { height = CURSES_MAX_HEIGHT; } if ( width > CURSES_MAX_WIDTH ) { width = CURSES_MAX_WIDTH; } globals->statusLine = height - MENU_WINDOW_HEIGHT - 1; menuWin = newwin( MENU_WINDOW_HEIGHT, width, height-MENU_WINDOW_HEIGHT, 0 ); nodelay(menuWin, 1); /* don't block on getch */ boardWin = newwin( height-MENU_WINDOW_HEIGHT, width, 0, 0 ); globals->menuWin = menuWin; globals->boardWin = boardWin; globals->mainWin = mainWin; } /* 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 static XP_Bool handleQuit( CursesAppGlobals* globals ) { globals->timeToExit = XP_TRUE; return XP_TRUE; } /* handleQuit */ static void checkAssignFocus( BoardCtxt* board ) { if ( OBJ_NONE == board_getFocusOwner(board) ) { board_focusChanged( board, OBJ_BOARD, XP_TRUE ); } } 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 &redo ); return XP_TRUE; } /* handleHint */ static XP_Bool handleCommit( CursesAppGlobals* globals ) { globals->doDraw = board_commitTurn( globals->cGlobals.game.board ); return XP_TRUE; } /* handleCommit */ static XP_Bool handleJuggle( CursesAppGlobals* globals ) { globals->doDraw = board_juggleTray( globals->cGlobals.game.board ); return XP_TRUE; } /* handleJuggle */ static XP_Bool handleHide( CursesAppGlobals* globals ) { XW_TrayVisState curState = board_getTrayVisState( globals->cGlobals.game.board ); if ( curState == TRAY_REVEALED ) { globals->doDraw = board_hideTray( globals->cGlobals.game.board ); } else { globals->doDraw = board_showTray( globals->cGlobals.game.board ); } return XP_TRUE; } /* handleJuggle */ #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 XP_Bool handleAltLeft( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_ALTLEFT ); } static XP_Bool handleAltRight( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_ALTRIGHT ); } static XP_Bool handleAltUp( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_ALTUP ); } static XP_Bool handleAltDown( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_ALTDOWN ); } static XP_Bool handleFlip( CursesAppGlobals* globals ) { globals->doDraw = board_flip( globals->cGlobals.game.board ); return XP_TRUE; } /* handleFlip */ static XP_Bool handleToggleValues( CursesAppGlobals* globals ) { globals->doDraw = board_toggle_showValues( globals->cGlobals.game.board ); return XP_TRUE; } /* handleToggleValues */ static XP_Bool handleBackspace( CursesAppGlobals* globals ) { XP_Bool handled; globals->doDraw = board_handleKey( globals->cGlobals.game.board, XP_CURSOR_KEY_DEL, &handled ); return XP_TRUE; } /* handleBackspace */ static XP_Bool handleUndo( CursesAppGlobals* globals ) { globals->doDraw = server_handleUndo( globals->cGlobals.game.server ); return XP_TRUE; } /* handleUndo */ static XP_Bool handleReplace( CursesAppGlobals* globals ) { globals->doDraw = board_replaceTiles( globals->cGlobals.game.board ); return XP_TRUE; } /* handleReplace */ #ifdef KEYBOARD_NAV static XP_Bool handleFocusKey( CursesAppGlobals* globals, XP_Key key ) { XP_Bool handled; XP_Bool draw; checkAssignFocus( globals->cGlobals.game.board ); draw = board_handleKey( globals->cGlobals.game.board, key, &handled ); if ( !handled ) { BoardObjectType nxt; BoardObjectType order[] = { OBJ_BOARD, OBJ_SCORE, OBJ_TRAY }; draw = linShiftFocus( &globals->cGlobals, key, order, &nxt ) || draw; if ( nxt != OBJ_NONE ) { changeMenuForFocus( globals, nxt ); } } globals->doDraw = draw || globals->doDraw; return XP_TRUE; } static XP_Bool handleLeft( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_LEFT ); } /* handleLeft */ static XP_Bool handleRight( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_RIGHT ); } /* handleRight */ static XP_Bool handleUp( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_UP ); } /* handleUp */ static XP_Bool handleDown( CursesAppGlobals* globals ) { return handleFocusKey( globals, XP_CURSOR_KEY_DOWN ); } /* handleDown */ #endif static void fmtMenuItem( const MenuList* item, char* buf, int maxLen ) { snprintf( buf, maxLen, "%s %s", item->keyDesc, item->desc ); } static void countMenuLines( 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 ) { 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, MenuList** menuLists, int nLines, int padding ) { short line = 0, col, i; int winMaxY, winMaxX; getmaxyx( win, winMaxY, winMaxX ); int maxColWidth = 0; if ( 0 == nLines ) { int ignore; countMenuLines( menuLists, winMaxX, padding, &nLines, &ignore ); } col = 0; for ( i = 0; NULL != menuLists[i]; ++i ) { 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 x, y; assert( signal == SIGWINCH ); endwin(); /* (*globals.drawMenu)( &globals ); */ getmaxyx( stdscr, y, x ); wresize( g_globals.mainWin, y-MENU_WINDOW_HEIGHT, x ); board_draw( g_globals.cGlobals.game.board ); } /* SIGWINCH_handler */ static void cursesListenOnSocket( CursesAppGlobals* globals, int newSock ) { XP_ASSERT( globals->fdCount+1 < FD_MAX ); XP_WARNF( "setting fd[%d] to %d", globals->fdCount, newSock ); globals->fdArray[globals->fdCount].fd = newSock; globals->fdArray[globals->fdCount].events = POLLIN; ++globals->fdCount; XP_LOGF( "listenOnSocket: there are now %d sources to poll", globals->fdCount ); } /* cursesListenOnSocket */ static void curses_stop_listening( CursesAppGlobals* globals, int sock ) { int count = globals->fdCount; int i; bool found = false; for ( i = 0; i < count; ++i ) { if ( globals->fdArray[i].fd == sock ) { found = true; } else if ( found ) { globals->fdArray[i-1].fd = globals->fdArray[i].fd; } } assert( found ); --globals->fdCount; } /* curses_stop_listening */ static void curses_socket_changed( void* closure, int oldSock, int newSock, void** XP_UNUSED(storage) ) { CursesAppGlobals* globals = (CursesAppGlobals*)closure; if ( oldSock != -1 ) { curses_stop_listening( globals, oldSock ); } if ( newSock != -1 ) { cursesListenOnSocket( globals, newSock ); } } /* curses_socket_changed */ static void curses_socket_acceptor( int listener, Acceptor func, CommonGlobals* cGlobals, void** XP_UNUSED(storage) ) { CursesAppGlobals* globals = (CursesAppGlobals*)cGlobals; XP_ASSERT( !cGlobals->acceptor || (func == cGlobals->acceptor) ); cGlobals->acceptor = func; globals->csInfo.server.serverSocket = listener; cursesListenOnSocket( globals, listener ); } #ifdef RELAY_HEARTBEAT static int figureTimeout( CursesAppGlobals* globals ) { int result = INFINITE_TIMEOUT; if ( globals->cGlobals.timerProcs[TIMER_HEARTBEAT] != 0 ) { XP_U32 now = util_getCurSeconds( globals->cGlobals.params->util ); XP_U32 then = globals->nextTimer; if ( now >= then ) { result = 0; } else { result = (then - now) * 1000; } } return result; } /* figureTimeout */ #else # define figureTimeout(g) INFINITE_TIMEOUT #endif /* * Ok, so this doesn't block yet.... */ static XP_Bool blocking_gotEvent( CursesAppGlobals* globals, int* ch ) { XP_Bool result = XP_FALSE; int numEvents; 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 RELAY_HEARTBEAT if ( !globals->cGlobals.params->noHeartbeat ) { linuxFireTimer( &globals->cGlobals, TIMER_HEARTBEAT ); } #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; /* XP_DEBUGF( "curses got a USER EVENT\n" ); */ (void)read(globals->fdArray[FD_TIMEEVT].fd, &ch, 1 ); } fdIndex = FD_FIRSTSOCKET; if ( numEvents > 0 && (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[256]; int nBytes; /* It's a normal data socket */ if ( 0 ) { #ifdef XWFEATURE_RELAY } else if ( globals->cGlobals.params->conType == COMMS_CONN_RELAY ) { nBytes = linux_relay_receive( &globals->cGlobals, buf, sizeof(buf) ); #endif #ifdef XWFEATURE_BLUETOOTH } else if ( globals->cGlobals.params->conType == COMMS_CONN_BT ) { nBytes = linux_bt_receive( globals->fdArray[fdIndex].fd, buf, sizeof(buf) ); #endif } else { XP_ASSERT( 0 ); } if ( nBytes != -1 ) { XWStreamCtxt* inboundS; struct sockaddr_in addr_sock; redraw = XP_FALSE; XP_STATUSF( "linuxReceive=>%d", nBytes ); inboundS = stream_from_msgbuf( &globals->cGlobals, buf, nBytes ); if ( !!inboundS ) { CommsAddrRec addrRec; XP_MEMSET( &addrRec, 0, sizeof(addrRec) ); addrRec.conType = COMMS_CONN_RELAY; addrRec.u.ip_relay.ipAddr = ntohl(addr_sock.sin_addr.s_addr); XP_LOGF( "captured incoming ip address: 0x%lx", addrRec.u.ip_relay.ipAddr ); if ( comms_checkIncomingStream( globals->cGlobals.game.comms, inboundS, &addrRec ) ) { XP_LOGF( "comms read port: %d", addrRec.u.ip_relay.port ); 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; } redraw = server_do( globals->cGlobals.game.server ) || redraw; if ( redraw ) { /* messages change a lot */ board_invalAll( globals->cGlobals.game.board ); board_draw( globals->cGlobals.game.board ); } } return result; } /* blocking_gotEvent */ 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 > 0xFF ) { XP_LOGF( "%s(%d): no mapping", __func__, key ); } break; } *kp = key; } static void drawMenuLargeOrSmall( CursesAppGlobals* globals, const MenuList* menuList ) { #ifdef CURSES_SMALL_SCREEN MenuList* lists[] = { g_rootMenuListShow, NULL }; #else MenuList* lists[] = { g_sharedMenuList, menuList, NULL }; #endif wclear( globals->menuWin ); drawMenuFromList( globals->menuWin, lists, 0, 0 ); wrefresh( globals->menuWin ); } static void changeMenuForFocus( CursesAppGlobals* globals, BoardObjectType focussed ) { #ifdef KEYBOARD_NAV if ( focussed == OBJ_TRAY ) { globals->menuList = g_trayMenuList; } else if ( focussed == OBJ_BOARD ) { globals->menuList = g_boardMenuList; } else if ( focussed == OBJ_SCORE ) { globals->menuList = g_scoreMenuList; } else { XP_ASSERT(0); } drawMenuLargeOrSmall( globals, globals->menuList ); #endif } /* changeMenuForFocus */ #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 VTableMgr* curses_util_getVTManager(XW_UtilCtxt* uc) { CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; return globals->cGlobals.params->vtMgr; } /* linux_util_getVTManager */ static XP_Bool curses_util_askPassword( XW_UtilCtxt* XP_UNUSED(uc), const XP_UCHAR* XP_UNUSED(name), XP_UCHAR* XP_UNUSED(buf), XP_U16* XP_UNUSED(len) ) { XP_WARNF( "curses_util_askPassword not implemented" ); return XP_FALSE; } /* curses_util_askPassword */ static void curses_util_yOffsetChange( XW_UtilCtxt* XP_UNUSED(uc), XP_U16 oldOffset, XP_U16 newOffset ) { if ( oldOffset != newOffset ) { XP_WARNF( "curses_util_yOffsetChange(%d,%d) not implemented", oldOffset, newOffset ); } } /* curses_util_yOffsetChange */ static XP_Bool curses_util_warnIllegalWord( XW_UtilCtxt* XP_UNUSED(uc), BadWordInfo* XP_UNUSED(bwi), XP_U16 XP_UNUSED(player), XP_Bool XP_UNUSED(turnLost) ) { XP_WARNF( "curses_util_warnIllegalWord not implemented" ); return XP_FALSE; } /* curses_util_warnIllegalWord */ #ifndef XWFEATURE_STANDALONE_ONLY static void cursesSendOnClose( XWStreamCtxt* stream, void* closure ) { XP_S16 result; CursesAppGlobals* globals = (CursesAppGlobals*)closure; XP_LOGF( "cursesSendOnClose called" ); result = comms_send( globals->cGlobals.game.comms, stream ); } /* cursesSendOnClose */ 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, uc->closure, channelNo, cursesSendOnClose ); return stream; } /* curses_util_makeStreamFromAddr */ #endif static void setupCursesUtilCallbacks( CursesAppGlobals* globals, XW_UtilCtxt* util ) { util->vtable->m_util_userError = curses_util_userError; util->vtable->m_util_getVTManager = curses_util_getVTManager; util->vtable->m_util_askPassword = curses_util_askPassword; util->vtable->m_util_yOffsetChange = curses_util_yOffsetChange; util->vtable->m_util_warnIllegalWord = curses_util_warnIllegalWord; #ifndef XWFEATURE_STANDALONE_ONLY util->vtable->m_util_makeStreamFromAddr = curses_util_makeStreamFromAddr; #endif util->vtable->m_util_userQuery = curses_util_userQuery; util->vtable->m_util_userPickTile = curses_util_userPickTile; util->vtable->m_util_trayHiddenChange = curses_util_trayHiddenChange; util->vtable->m_util_notifyGameOver = curses_util_notifyGameOver; util->vtable->m_util_hiliteCell = curses_util_hiliteCell; util->vtable->m_util_engineProgressCallback = curses_util_engineProgressCallback; util->vtable->m_util_setTimer = curses_util_setTimer; util->vtable->m_util_requestTime = curses_util_requestTime; util->closure = globals; } /* setupCursesUtilCallbacks */ #ifndef XWFEATURE_STANDALONE_ONLY static void sendOnClose( XWStreamCtxt* stream, void* closure ) { CursesAppGlobals* globals = closure; XP_LOGF( "curses sendOnClose called" ); XP_ASSERT( !!globals->cGlobals.game.comms ); comms_send( globals->cGlobals.game.comms, stream ); } /* sendOnClose */ #endif static CursesMenuHandler getHandlerForKey( 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, 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 ) { XP_U16 cellWidth, cellHt, scoreLeft, scoreWidth; BoardCtxt* board = globals->cGlobals.game.board; int remWidth = width; board_setPos( board, BOARD_OFFSET, BOARD_OFFSET, XP_FALSE ); cellWidth = CURSES_CELL_WIDTH; cellHt = CURSES_CELL_HT; board_setScale( board, cellWidth, cellHt ); scoreLeft = (cellWidth * MAX_COLS);// + BOARD_SCORE_PADDING; remWidth -= cellWidth * MAX_COLS; /* If the scoreboard will right of the board, put it there. Otherwise try to fit it below the boards. */ int tileWidth = 3; int trayWidth = (tileWidth*MAX_TRAY_TILES); int trayLeft = scoreLeft; int trayTop; int trayHt = 4; if ( trayWidth < remWidth ) { trayLeft += XP_MIN(remWidth - trayWidth, BOARD_SCORE_PADDING ); trayTop = 8; } else { trayLeft = BOARD_OFFSET; trayTop = BOARD_OFFSET + (cellHt * MAX_ROWS); if ( trayTop + trayHt > height ) { trayHt = height - trayTop; } } board_setTrayLoc( board, trayLeft, trayTop, (3*MAX_TRAY_TILES)+1, trayHt, 1 ); scoreWidth = remWidth; if ( scoreWidth > 45 ) { scoreWidth = 45; scoreLeft += (remWidth - scoreWidth) / 2; } board_setScoreboardLoc( board, scoreLeft, 1, scoreWidth, 5, /*4 players + rem*/ XP_FALSE ); /* no divider -- yet */ /* board_setTrayVisible( globals.board, XP_TRUE, XP_FALSE ); */ board_invalAll( board ); } /* positionSizeStuff */ void cursesmain( XP_Bool isServer, LaunchParams* params ) { int piperesult; DictionaryCtxt* dict; XP_U16 gameID; int width, height; memset( &g_globals, 0, sizeof(g_globals) ); g_globals.amServer = isServer; g_globals.cGlobals.params = params; #ifdef XWFEATURE_RELAY g_globals.cGlobals.socket = -1; #endif g_globals.cGlobals.socketChanged = curses_socket_changed; g_globals.cGlobals.socketChangedClosure = &g_globals; g_globals.cGlobals.addAcceptor = curses_socket_acceptor; g_globals.cp.showBoardArrow = XP_TRUE; g_globals.cp.showRobotScores = params->showRobotScores; dict = params->dict; setupCursesUtilCallbacks( &g_globals, params->util ); #ifdef XWFEATURE_RELAY if ( params->conType == COMMS_CONN_RELAY ) { g_globals.cGlobals.defaultServerName = params->connInfo.relay.relayName; } #endif cursesListenOnSocket( &g_globals, 0 ); /* stdin */ piperesult = pipe( g_globals.timepipe ); XP_ASSERT( piperesult == 0 ); /* reader pipe */ cursesListenOnSocket( &g_globals, g_globals.timepipe[0] ); signal( SIGWINCH, SIGWINCH_handler ); initCurses( &g_globals ); getmaxyx( g_globals.boardWin, height, width ); g_globals.draw = (struct CursesDrawCtx*) cursesDrawCtxtMake( g_globals.boardWin ); gameID = (XP_U16)util_getCurSeconds( g_globals.cGlobals.params->util ); game_makeNewGame( MEMPOOL &g_globals.cGlobals.game, ¶ms->gi, params->util, (DrawCtx*)g_globals.draw, gameID, &g_globals.cp, LINUX_SEND, IF_CH(linux_reset) &g_globals ); #ifndef XWFEATURE_STANDALONE_ONLY if ( g_globals.cGlobals.game.comms ) { CommsAddrRec addr; if ( 0 ) { # ifdef XWFEATURE_RELAY } else if ( params->conType == COMMS_CONN_RELAY ) { addr.conType = COMMS_CONN_RELAY; addr.u.ip_relay.ipAddr = 0; /* ??? */ addr.u.ip_relay.port = params->connInfo.relay.defaultSendPort; XP_STRNCPY( addr.u.ip_relay.hostName, params->connInfo.relay.relayName, sizeof(addr.u.ip_relay.hostName) - 1 ); XP_STRNCPY( addr.u.ip_relay.cookie, params->connInfo.relay.cookie, sizeof(addr.u.ip_relay.cookie) - 1 ); # endif # ifdef XWFEATURE_BLUETOOTH } else if ( params->conType == COMMS_CONN_BT ) { addr.conType = COMMS_CONN_BT; XP_ASSERT( sizeof(addr.u.bt.btAddr) >= sizeof(params->connInfo.bt.hostAddr)); XP_MEMCPY( &addr.u.bt.btAddr, ¶ms->connInfo.bt.hostAddr, sizeof(params->connInfo.bt.hostAddr) ); # endif } comms_setAddr( g_globals.cGlobals.game.comms, &addr ); } #endif model_setDictionary( g_globals.cGlobals.game.model, params->dict ); positionSizeStuff( &g_globals, width, height ); #ifndef XWFEATURE_STANDALONE_ONLY /* send any events that need to get off before the event loop begins */ if ( !isServer ) { if ( 1 /* stream_open( params->info.clientInfo.stream ) */) { server_initClientConnection( g_globals.cGlobals.game.server, mem_stream_make( MEMPOOL params->vtMgr, &g_globals, (XP_PlayerAddr)0, sendOnClose ) ); } else { cursesUserError( &g_globals, "Unable to open connection to server"); exit( 0 ); } } #endif server_do( g_globals.cGlobals.game.server ); g_globals.menuList = g_boardMenuList; drawMenuLargeOrSmall( &g_globals, g_boardMenuList ); board_draw( g_globals.cGlobals.game.board ); while ( !g_globals.timeToExit ) { int ch; if ( blocking_gotEvent( &g_globals, &ch ) ) { remapKey( &ch ); if ( #ifdef CURSES_SMALL_SCREEN handleKeyEvent( &g_globals, g_rootMenuListShow, ch ) || #endif handleKeyEvent( &g_globals, g_globals.menuList, ch ) || handleKeyEvent( &g_globals, g_sharedMenuList, ch ) || passKeyToBoard( &g_globals, ch ) ) { if ( g_globals.doDraw ) { board_draw( g_globals.cGlobals.game.board ); g_globals.doDraw = XP_FALSE; } } } } endwin(); } /* cursesmain */ #endif /* PLATFORM_NCURSES */