mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-16 15:41:16 +01:00
9239d34f19
info: when the device is a client and is starting a new game, we want to start presenting only the local players. So load them first, and reduce nPlayers down to the count of current local players. Works well, but can probably be simplified.
466 lines
14 KiB
C
466 lines
14 KiB
C
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
/*
|
|
* Copyright 1997 - 2006 by Eric House (xwords@eehouse.org). All rights
|
|
* reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
|
|
#include "nwgamest.h"
|
|
#include "strutils.h"
|
|
#include "LocalizedStrIncludes.h"
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
#define NG_NUM_COLS ((XP_U16)(NG_COL_PASSWD+1))
|
|
|
|
struct NewGameCtx {
|
|
NewGameEnableColProc enableColProc;
|
|
NewGameEnableAttrProc enableAttrProc;
|
|
NewGameSetColProc setColProc;
|
|
NewGameGetColProc getColProc;
|
|
NewGameSetAttrProc setAttrProc;
|
|
XW_UtilCtxt* util;
|
|
void* closure;
|
|
|
|
/* Palm needs to store cleartext passwords separately in order to
|
|
store '***' in the visible field */
|
|
NewGameEnable enabled[NG_NUM_COLS][MAX_NUM_PLAYERS];
|
|
XP_U16 nPlayers;
|
|
Connectedness role;
|
|
XP_Bool isNewGame;
|
|
NewGameEnable juggleEnabled;
|
|
|
|
MPSLOT
|
|
};
|
|
|
|
static void enableOne( NewGameCtx* ngc, XP_U16 player, NewGameColumn col,
|
|
NewGameEnable enable, XP_Bool force );
|
|
static void adjustAllRows( NewGameCtx* ngc, XP_Bool force );
|
|
static void adjustOneRow( NewGameCtx* ngc, XP_U16 player, XP_Bool force );
|
|
static void setRoleStrings( NewGameCtx* ngc );
|
|
static void considerEnableJuggle( NewGameCtx* ngc );
|
|
static void storePlayer( NewGameCtx* ngc, XP_U16 player, LocalPlayer* lp );
|
|
static void loadPlayer( NewGameCtx* ngc, XP_U16 player,
|
|
const LocalPlayer* lp );
|
|
|
|
NewGameCtx*
|
|
newg_make( MPFORMAL XP_Bool isNewGame,
|
|
XW_UtilCtxt* util,
|
|
NewGameEnableColProc enableColProc,
|
|
NewGameEnableAttrProc enableAttrProc,
|
|
NewGameGetColProc getColProc, NewGameSetColProc setColProc,
|
|
NewGameSetAttrProc setAttrProc, void* closure )
|
|
{
|
|
NewGameCtx* result = XP_MALLOC( mpool, sizeof(*result) );
|
|
XP_MEMSET( result, 0, sizeof(*result) );
|
|
|
|
result->enableColProc = enableColProc;
|
|
result->enableAttrProc = enableAttrProc;
|
|
result->setColProc = setColProc;
|
|
result->getColProc = getColProc;
|
|
result->setAttrProc = setAttrProc;
|
|
result->closure = closure;
|
|
result->isNewGame = isNewGame;
|
|
result->util = util;
|
|
MPASSIGN(result->mpool, mpool);
|
|
|
|
return result;
|
|
} /* newg_make */
|
|
|
|
void
|
|
newg_destroy( NewGameCtx* ngc )
|
|
{
|
|
XP_FREE( ngc->mpool, ngc );
|
|
} /* newg_destroy */
|
|
|
|
void
|
|
newg_load( NewGameCtx* ngc, const CurGameInfo* gi )
|
|
{
|
|
void* closure = ngc->closure;
|
|
NGValue value;
|
|
XP_U16 nPlayers, nShown;
|
|
XP_S16 i;
|
|
Connectedness role;
|
|
XP_Bool localOnly;
|
|
XP_Bool shown[MAX_NUM_PLAYERS] = { XP_FALSE, XP_FALSE, XP_FALSE, XP_FALSE};
|
|
|
|
ngc->role = role = gi->serverRole;
|
|
localOnly = role == SERVER_ISCLIENT;
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
value.ng_role = role;
|
|
(*ngc->setAttrProc)( closure, NG_ATTR_ROLE, value );
|
|
(*ngc->enableAttrProc)( closure, NG_ATTR_ROLE, ngc->isNewGame?
|
|
NGEnableEnabled : NGEnableDisabled );
|
|
#endif
|
|
|
|
nPlayers = gi->nPlayers;
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
if ( localOnly ) {
|
|
for ( i = nPlayers - 1; i >= 0; --i ) {
|
|
if ( !gi->players[i].isLocal ) {
|
|
--nPlayers;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
ngc->nPlayers = nPlayers;
|
|
value.ng_u16 = ngc->nPlayers;
|
|
(*ngc->setAttrProc)( closure, NG_ATTR_NPLAYERS, value );
|
|
(*ngc->enableAttrProc)( closure, NG_ATTR_NPLAYERS, ngc->isNewGame?
|
|
NGEnableEnabled : NGEnableDisabled );
|
|
|
|
setRoleStrings( ngc );
|
|
considerEnableJuggle( ngc );
|
|
|
|
/* Load local players first */
|
|
nShown = 0;
|
|
do {
|
|
for ( i = 0; i < MAX_NUM_PLAYERS; ++i ) {
|
|
const LocalPlayer* lp = &gi->players[i];
|
|
if ( shown[i] ) {
|
|
/* already got it */
|
|
} else if ( !localOnly || lp->isLocal ) {
|
|
shown[i] = XP_TRUE;
|
|
loadPlayer( ngc, nShown++, lp );
|
|
} /* else skip it */
|
|
}
|
|
XP_ASSERT( localOnly || nShown == MAX_NUM_PLAYERS );
|
|
localOnly = XP_FALSE; /* for second pass */
|
|
} while ( nShown < MAX_NUM_PLAYERS );
|
|
|
|
adjustAllRows( ngc, XP_TRUE );
|
|
} /* newg_load */
|
|
|
|
typedef struct NGCopyClosure {
|
|
XP_U16 player;
|
|
NewGameColumn col;
|
|
NewGameCtx* ngc;
|
|
LocalPlayer* lp;
|
|
} NGCopyClosure;
|
|
|
|
static void
|
|
cpToLP( NGValue value, const void* cbClosure )
|
|
{
|
|
NGCopyClosure* cpcl = (NGCopyClosure*)cbClosure;
|
|
LocalPlayer* lp = cpcl->lp;
|
|
XP_UCHAR** strAddr = NULL;
|
|
|
|
switch ( cpcl->col ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_COL_REMOTE:
|
|
lp->isLocal = !value.ng_bool;
|
|
break;
|
|
#endif
|
|
case NG_COL_NAME:
|
|
strAddr = &lp->name;
|
|
break;
|
|
case NG_COL_PASSWD:
|
|
strAddr = &lp->password;
|
|
break;
|
|
case NG_COL_ROBOT:
|
|
lp->isRobot = value.ng_bool;
|
|
break;
|
|
}
|
|
|
|
if ( !!strAddr ) {
|
|
/* This is leaking!!! But doesn't leak if am playing via IR, at least
|
|
in the simulator. */
|
|
replaceStringIfDifferent( cpcl->ngc->mpool, strAddr,
|
|
value.ng_cp );
|
|
}
|
|
} /* cpToLP */
|
|
|
|
void
|
|
newg_store( NewGameCtx* ngc, CurGameInfo* gi )
|
|
{
|
|
XP_U16 player;
|
|
|
|
gi->nPlayers = ngc->nPlayers;
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
gi->serverRole = ngc->role;
|
|
#endif
|
|
|
|
for ( player = 0; player < MAX_NUM_PLAYERS; ++player ) {
|
|
storePlayer( ngc, player, &gi->players[player] );
|
|
}
|
|
} /* newg_store */
|
|
|
|
void
|
|
newg_colChanged( NewGameCtx* ngc, XP_U16 player )
|
|
{
|
|
/* Sometimes we'll get this notification for inactive rows, e.g. when
|
|
setting default values. */
|
|
if ( player < ngc->nPlayers ) {
|
|
adjustOneRow( ngc, player, XP_FALSE );
|
|
}
|
|
}
|
|
|
|
void
|
|
newg_attrChanged( NewGameCtx* ngc, NewGameAttr attr, NGValue value )
|
|
{
|
|
if ( attr == NG_ATTR_NPLAYERS ) {
|
|
ngc->nPlayers = value.ng_u16;
|
|
considerEnableJuggle( ngc );
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
} else if ( NG_ATTR_ROLE == attr ) {
|
|
ngc->role = value.ng_role;
|
|
setRoleStrings( ngc );
|
|
#endif
|
|
} else {
|
|
XP_ASSERT( 0 );
|
|
}
|
|
adjustAllRows( ngc, XP_FALSE );
|
|
}
|
|
|
|
typedef struct DeepValue {
|
|
NGValue value;
|
|
NewGameColumn col;
|
|
MPSLOT
|
|
} DeepValue;
|
|
|
|
static void
|
|
deepCopy( NGValue value, const void* closure )
|
|
{
|
|
DeepValue* dvp = (DeepValue*)closure;
|
|
switch ( dvp->col ) {
|
|
case NG_COL_ROBOT:
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_COL_REMOTE:
|
|
#endif
|
|
dvp->value.ng_bool = value.ng_bool;
|
|
break;
|
|
case NG_COL_NAME:
|
|
case NG_COL_PASSWD:
|
|
dvp->value.ng_cp = copyString( dvp->mpool, value.ng_cp );
|
|
break;
|
|
}
|
|
}
|
|
|
|
XP_Bool
|
|
newg_juggle( NewGameCtx* ngc )
|
|
{
|
|
XP_Bool changed = XP_FALSE;
|
|
XP_U16 nPlayers = ngc->nPlayers;
|
|
|
|
if ( nPlayers > 1 ) {
|
|
LocalPlayer tmpPlayers[MAX_NUM_PLAYERS];
|
|
XP_U16 pos[MAX_NUM_PLAYERS];
|
|
XP_U16 player;
|
|
|
|
/* Get a randomly juggled array of numbers 0..nPlayers-1. Then the
|
|
number at pos[n] inicates where the entry currently at n should
|
|
be. */
|
|
changed = randIntArray( pos, nPlayers );
|
|
if ( changed ) {
|
|
|
|
/* Deep-copy off to tmp storage. But skip lines that won't be moved
|
|
in the juggle. */
|
|
XP_MEMSET( &tmpPlayers, 0, sizeof(tmpPlayers) );
|
|
for ( player = 0; player < nPlayers; ++player ) {
|
|
if ( player != pos[player] ) {
|
|
storePlayer( ngc, player, &tmpPlayers[player] );
|
|
}
|
|
}
|
|
|
|
for ( player = 0; player < nPlayers; ++player ) {
|
|
if ( player != pos[player] ) {
|
|
LocalPlayer* lp = &tmpPlayers[player];
|
|
XP_U16 dest = pos[player];
|
|
|
|
loadPlayer( ngc, dest, lp );
|
|
|
|
if ( !!lp->name ) {
|
|
XP_FREE( ngc->mpool, lp->name );
|
|
}
|
|
if ( !!lp->password ) {
|
|
XP_FREE( ngc->mpool, lp->password );
|
|
}
|
|
|
|
adjustOneRow( ngc, dest, XP_FALSE );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return changed;
|
|
} /* newg_juggle */
|
|
|
|
static void
|
|
enableOne( NewGameCtx* ngc, XP_U16 player, NewGameColumn col,
|
|
NewGameEnable enable, XP_Bool force )
|
|
{
|
|
NewGameEnable* esp = &ngc->enabled[col][player];
|
|
if ( force || (*esp != enable) ) {
|
|
(*ngc->enableColProc)( ngc->closure, player, col, enable );
|
|
}
|
|
*esp = enable;
|
|
} /* enableOne */
|
|
|
|
static void
|
|
adjustAllRows( NewGameCtx* ngc, XP_Bool force )
|
|
{
|
|
XP_U16 player;
|
|
for ( player = 0; player < MAX_NUM_PLAYERS; ++player ) {
|
|
adjustOneRow( ngc, player, force );
|
|
}
|
|
} /* adjustAllRows */
|
|
|
|
static void
|
|
adjustOneRow( NewGameCtx* ngc, XP_U16 player, XP_Bool force )
|
|
{
|
|
NewGameEnable enable[NG_NUM_COLS];
|
|
NewGameColumn col;
|
|
XP_Bool isLocal = XP_TRUE;
|
|
DeepValue dValue;
|
|
|
|
for ( col = 0; col < NG_NUM_COLS; ++col ) {
|
|
enable[col] = NGEnableHidden;
|
|
}
|
|
|
|
/* If there aren't this many players, all are disabled */
|
|
if ( player >= ngc->nPlayers ) {
|
|
/* do nothing: all are hidden above */
|
|
} else {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
/* If standalone or client, remote is hidden. If server but not
|
|
new game, it's disabled */
|
|
if ( ngc->role == SERVER_ISSERVER ) {
|
|
if ( ngc->isNewGame ) {
|
|
enable[NG_COL_REMOTE] = NGEnableEnabled;
|
|
} else {
|
|
enable[NG_COL_REMOTE] = NGEnableDisabled;
|
|
}
|
|
dValue.col = NG_COL_REMOTE;
|
|
(*ngc->getColProc)( ngc->closure, player, NG_COL_REMOTE,
|
|
deepCopy, &dValue );
|
|
isLocal = !dValue.value.ng_bool;
|
|
}
|
|
#endif
|
|
|
|
/* If remote is enabled and set, then if it's a new game all else is
|
|
hidden. But if it's not a new game, they're disabled. Password is
|
|
always hidden if robot is set. */
|
|
if ( isLocal ) {
|
|
/* No changing name or robotness since they're sent to remote
|
|
host. */
|
|
enable[NG_COL_NAME] = NGEnableEnabled;
|
|
enable[NG_COL_ROBOT] = NGEnableEnabled;
|
|
|
|
dValue.col = NG_COL_ROBOT;
|
|
(*ngc->getColProc)( ngc->closure, player, NG_COL_ROBOT, deepCopy,
|
|
&dValue );
|
|
if ( !dValue.value.ng_bool ) {
|
|
/* It is's a robot, leave it hidden */
|
|
enable[NG_COL_PASSWD] = NGEnableEnabled;
|
|
}
|
|
|
|
} else {
|
|
if ( ngc->isNewGame ) {
|
|
/* leave 'em hidden */
|
|
} else {
|
|
enable[NG_COL_NAME] = NGEnableDisabled;
|
|
enable[NG_COL_ROBOT] = NGEnableDisabled;
|
|
/* leave passwd hidden */
|
|
}
|
|
}
|
|
}
|
|
|
|
for ( col = 0; col < NG_NUM_COLS; ++col ) {
|
|
enableOne( ngc, player, col, enable[col], force );
|
|
}
|
|
} /* adjustOneRow */
|
|
|
|
static void
|
|
setRoleStrings( NewGameCtx* ngc )
|
|
{
|
|
XP_U16 strID;
|
|
NGValue value;
|
|
void* closure = ngc->closure;
|
|
/* Tell client to set/change players label text, and also to add remote
|
|
checkbox column header if required. */
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
(*ngc->enableAttrProc)( closure, NG_ATTR_REMHEADER,
|
|
ngc->role == SERVER_ISSERVER?
|
|
NGEnableEnabled : NGEnableHidden );
|
|
#endif
|
|
|
|
if ( 0 ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
} else if ( ngc->role == SERVER_ISCLIENT ) {
|
|
strID = STR_LOCALPLAYERS;
|
|
#endif
|
|
} else {
|
|
strID = STR_TOTALPLAYERS;
|
|
}
|
|
|
|
value.ng_cp = util_getUserString( ngc->util, strID );
|
|
(*ngc->setAttrProc)( closure, NG_ATTR_NPLAYHEADER, value );
|
|
} /* setRoleStrings */
|
|
|
|
static void
|
|
considerEnableJuggle( NewGameCtx* ngc )
|
|
{
|
|
NewGameEnable newEnable;
|
|
newEnable = (ngc->isNewGame && ngc->nPlayers > 1)?
|
|
NGEnableEnabled : NGEnableDisabled;
|
|
|
|
if ( newEnable != ngc->juggleEnabled ) {
|
|
(*ngc->enableAttrProc)( ngc->closure, NG_ATTR_CANJUGGLE, newEnable );
|
|
ngc->juggleEnabled = newEnable;
|
|
}
|
|
} /* considerEnableJuggle */
|
|
|
|
static void
|
|
storePlayer( NewGameCtx* ngc, XP_U16 player, LocalPlayer* lp )
|
|
{
|
|
void* closure = ngc->closure;
|
|
NGCopyClosure cpcl;
|
|
cpcl.player = player;
|
|
cpcl.ngc = ngc;
|
|
cpcl.lp = lp;
|
|
|
|
for ( cpcl.col = 0; cpcl.col < NG_NUM_COLS; ++cpcl.col ) {
|
|
(*ngc->getColProc)( closure, cpcl.player, cpcl.col,
|
|
cpToLP, &cpcl );
|
|
}
|
|
}
|
|
|
|
static void
|
|
loadPlayer( NewGameCtx* ngc, XP_U16 player, const LocalPlayer* lp )
|
|
{
|
|
NGValue value;
|
|
void* closure = ngc->closure;
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
value.ng_bool = !lp->isLocal;
|
|
(*ngc->setColProc)(closure, player, NG_COL_REMOTE, value );
|
|
#endif
|
|
value.ng_cp = lp->name;
|
|
(*ngc->setColProc)(closure, player, NG_COL_NAME, value );
|
|
|
|
value.ng_cp = lp->password;
|
|
(*ngc->setColProc)(closure, player, NG_COL_PASSWD, value );
|
|
|
|
value.ng_bool = lp->isRobot;
|
|
(*ngc->setColProc)(closure, player, NG_COL_ROBOT, value );
|
|
}
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|