Warn user, and refuse to continue, when new game is as host and has no remote players. API

changes, so other platforms won't compile until changed.
This commit is contained in:
ehouse 2008-01-04 14:47:52 +00:00
parent a25de221e9
commit dfee56609a
5 changed files with 55 additions and 10 deletions

View file

@ -43,7 +43,7 @@ struct NewGameCtx {
XP_TriEnable enabled[NG_NUM_COLS][MAX_NUM_PLAYERS]; XP_TriEnable enabled[NG_NUM_COLS][MAX_NUM_PLAYERS];
XP_U16 nPlayersShown; /* real nPlayers lives in gi */ XP_U16 nPlayersShown; /* real nPlayers lives in gi */
XP_U16 nPlayersTotal; /* used only until changedNPlayers set */ XP_U16 nPlayersTotal; /* used only until changedNPlayers set */
XP_U16 nLocalPlayers; XP_U16 nLocalPlayers; /* not changed except in ngc_load */
DeviceRole role; DeviceRole role;
XP_Bool isNewGame; XP_Bool isNewGame;
XP_Bool changedNPlayers; XP_Bool changedNPlayers;
@ -62,6 +62,7 @@ static void considerEnableJuggle( NewGameCtx* ngc );
static void storePlayer( NewGameCtx* ngc, XP_U16 player, LocalPlayer* lp ); static void storePlayer( NewGameCtx* ngc, XP_U16 player, LocalPlayer* lp );
static void loadPlayer( NewGameCtx* ngc, XP_U16 player, static void loadPlayer( NewGameCtx* ngc, XP_U16 player,
const LocalPlayer* lp ); const LocalPlayer* lp );
static XP_Bool checkConsistent( NewGameCtx* ngc, XP_Bool warnUser );
NewGameCtx* NewGameCtx*
newg_make( MPFORMAL XP_Bool isNewGame, newg_make( MPFORMAL XP_Bool isNewGame,
@ -201,19 +202,23 @@ cpToLP( NGValue value, const void* cbClosure )
} }
} /* cpToLP */ } /* cpToLP */
void XP_Bool
newg_store( NewGameCtx* ngc, CurGameInfo* gi ) newg_store( NewGameCtx* ngc, CurGameInfo* gi, XP_Bool warn )
{ {
XP_U16 player; XP_U16 player;
XP_Bool consistent = checkConsistent( ngc, warn );
gi->nPlayers = ngc->nPlayersShown; if ( consistent ) {
gi->nPlayers = ngc->nPlayersShown;
#ifndef XWFEATURE_STANDALONE_ONLY #ifndef XWFEATURE_STANDALONE_ONLY
gi->serverRole = ngc->role; gi->serverRole = ngc->role;
#endif #endif
for ( player = 0; player < MAX_NUM_PLAYERS; ++player ) { for ( player = 0; player < MAX_NUM_PLAYERS; ++player ) {
storePlayer( ngc, player, &gi->players[player] ); storePlayer( ngc, player, &gi->players[player] );
}
} }
return consistent;
} /* newg_store */ } /* newg_store */
void void
@ -320,6 +325,32 @@ newg_juggle( NewGameCtx* ngc )
return changed; return changed;
} /* newg_juggle */ } /* newg_juggle */
static XP_Bool
checkConsistent( NewGameCtx* ngc, XP_Bool warnUser )
{
XP_Bool consistent;
XP_U16 i;
/* If ISSERVER, make sure there's at least one non-local player. */
consistent = ngc->role != SERVER_ISSERVER;
for ( i = 0; !consistent && i < ngc->nPlayersShown; ++i ) {
DeepValue dValue;
dValue.col = NG_COL_REMOTE;
(*ngc->getColProc)( ngc->closure, i, NG_COL_REMOTE,
deepCopy, &dValue );
if ( dValue.value.ng_bool ) {
consistent = XP_TRUE;
}
}
if ( !consistent && warnUser ) {
util_userError( ngc->util, ERR_REG_SERVER_SANS_REMOTE );
}
/* Add other consistency checks, and error messages, here. */
return consistent;
} /* checkConsistent */
static void static void
enableOne( NewGameCtx* ngc, XP_U16 player, NewGameColumn col, enableOne( NewGameCtx* ngc, XP_U16 player, NewGameColumn col,
XP_TriEnable enable, XP_Bool force ) XP_TriEnable enable, XP_Bool force )

View file

@ -98,7 +98,7 @@ NewGameCtx* newg_make( MPFORMAL XP_Bool isNewGame,
void newg_destroy( NewGameCtx* ngc ); void newg_destroy( NewGameCtx* ngc );
void newg_load( NewGameCtx* ngc, const CurGameInfo* gi ); void newg_load( NewGameCtx* ngc, const CurGameInfo* gi );
void newg_store( NewGameCtx* ngc, CurGameInfo* gi ); XP_Bool newg_store( NewGameCtx* ngc, CurGameInfo* gi, XP_Bool warn );
void newg_colChanged( NewGameCtx* ngc, XP_U16 player ); void newg_colChanged( NewGameCtx* ngc, XP_U16 player );
void newg_attrChanged( NewGameCtx* ngc, NewGameAttr attr, void newg_attrChanged( NewGameCtx* ngc, NewGameAttr attr,

View file

@ -48,6 +48,7 @@ typedef enum {
ERR_NO_PEEK_REMOTE_TILES, ERR_NO_PEEK_REMOTE_TILES,
ERR_REG_UNEXPECTED_USER, /* server asked to register too many remote ERR_REG_UNEXPECTED_USER, /* server asked to register too many remote
users */ users */
ERR_REG_SERVER_SANS_REMOTE,
#endif #endif
ERR_CANT_TRADE_MID_MOVE, ERR_CANT_TRADE_MID_MOVE,
/* ERR_CANT_ENGINE_MID_MOVE, */ /* ERR_CANT_ENGINE_MID_MOVE, */

View file

@ -498,8 +498,13 @@ newGameDialog( GtkAppGlobals* globals, XP_Bool isNewGame )
gtk_main(); gtk_main();
if ( !state.cancelled && !state.revert ) { if ( !state.cancelled && !state.revert ) {
newg_store( state.newGameCtxt, &globals->cGlobals.params->gi ); if ( newg_store( state.newGameCtxt, &globals->cGlobals.params->gi,
globals->cGlobals.params->gi.boardSize = state.nCols; XP_TRUE ) ) {
globals->cGlobals.params->gi.boardSize = state.nCols;
} else {
/* Do it again if we warned user of inconsistency. */
state.revert = XP_TRUE;
}
} }
gtk_widget_destroy( dialog ); gtk_widget_destroy( dialog );

View file

@ -166,6 +166,8 @@ linux_getErrString( UtilErrID id, XP_Bool* silent )
case ERR_NO_PEEK_ROBOT_TILES: case ERR_NO_PEEK_ROBOT_TILES:
message = "No peeking at the robot's tiles!"; message = "No peeking at the robot's tiles!";
break; break;
#ifndef XWFEATURE_STANDALONE_ONLY
case ERR_NO_PEEK_REMOTE_TILES: case ERR_NO_PEEK_REMOTE_TILES:
message = "No peeking at remote players' tiles!"; message = "No peeking at remote players' tiles!";
break; break;
@ -176,6 +178,12 @@ linux_getErrString( UtilErrID id, XP_Bool* silent )
message = "Conflict between Host and Guest dictionaries; Host wins."; message = "Conflict between Host and Guest dictionaries; Host wins.";
XP_WARNF( "GTK may have problems here." ); XP_WARNF( "GTK may have problems here." );
break; break;
case ERR_REG_SERVER_SANS_REMOTE:
message = "At least one player must be marked remote for a game "
"started as Host.";
break;
#endif
case ERR_CANT_UNDO_TILEASSIGN: case ERR_CANT_UNDO_TILEASSIGN:
message = "Tile assignment can't be undone."; message = "Tile assignment can't be undone.";
break; break;