mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-04 23:02:02 +01:00
ba4f91d9e0
XWFEATURE_BLUETOOTH. The goal is to be able to build to support bluetooth only, or relay/ip only, e.g. for a palm bluetooth beta. Seems to work.
945 lines
27 KiB
C
945 lines
27 KiB
C
/* -*-mode: C; fill-column: 77; c-basic-offset: 4; -*- */
|
|
/*
|
|
* Copyright 1999 - 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 <PalmTypes.h>
|
|
#include <Form.h>
|
|
#include <List.h>
|
|
#include <Chars.h> /* for nextFieldChr */
|
|
#include <Graffiti.h> /* for GrfSetState */
|
|
#include <Event.h>
|
|
#ifdef HS_DUO_SUPPORT
|
|
# include <Hs.h>
|
|
#endif
|
|
|
|
#include "callback.h"
|
|
#include "comtypes.h"
|
|
#include "palmmain.h"
|
|
#include "comms.h"
|
|
#include "strutils.h"
|
|
#include "newgame.h"
|
|
#include "xwords4defines.h"
|
|
#include "dictui.h"
|
|
#include "palmdict.h"
|
|
#include "palmutil.h"
|
|
#include "palmir.h"
|
|
#include "prefsdlg.h"
|
|
#include "connsdlg.h"
|
|
|
|
static void handlePasswordTrigger( PalmAppGlobals* globals,
|
|
UInt16 controlID );
|
|
static void updatePlayerInfo( PalmAppGlobals* globals );
|
|
static XP_Bool tryFieldNavigationKey( XP_U16 key );
|
|
static void loadNewGameState( PalmAppGlobals* globals );
|
|
static void unloadNewGameState( PalmAppGlobals* globals );
|
|
static void setNameThatFits( PalmNewGameState* state );
|
|
#ifdef HS_DUO_SUPPORT
|
|
static XP_Bool tryDuoRockerKey( PalmAppGlobals* globals,XP_U16 key );
|
|
static XP_Bool considerGadgetFocus( PalmNewGameState* state, EventType* event );
|
|
#else
|
|
# define tryDuoRockerKey(g,key) XP_FALSE
|
|
#endif
|
|
|
|
static XP_Bool onDisabledList( const PalmNewGameState* state,
|
|
XP_U16 controlID );
|
|
static void modDisabledList( PalmNewGameState* state, XP_U16 controlID,
|
|
NewGameEnable enable );
|
|
|
|
static void palmEnableColProc( void* closure, XP_U16 player,
|
|
NewGameColumn col, NewGameEnable enable );
|
|
static void palmEnableAttrProc( void* closure, NewGameAttr attr,
|
|
NewGameEnable enable );
|
|
static void palmGetColProc( void* closure, XP_U16 player, NewGameColumn col,
|
|
NgCpCallbk cpcb, const void* cbClosure );
|
|
static void palmSetColProc( void* closure, XP_U16 player, NewGameColumn col,
|
|
const NGValue value );
|
|
static void palmSetAttrProc( void* closure, NewGameAttr attr,
|
|
const NGValue value );
|
|
static void handleRobotChanged( PalmNewGameState* state, XP_U16 controlID,
|
|
XP_Bool on );
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
static void handleRemoteChanged( PalmNewGameState* state, XP_U16 controlID,
|
|
XP_Bool on );
|
|
static Boolean checkHiliteGadget(PalmAppGlobals* globals, EventType* event,
|
|
PalmNewGameState* state );
|
|
static void drawConnectGadgets( PalmAppGlobals* globals );
|
|
static void changeGadgetHilite( PalmAppGlobals* globals, UInt16 hiliteID );
|
|
|
|
#else
|
|
# define checkHiliteGadget(globals, event, state) XP_FALSE
|
|
# define drawConnectGadgets( globals )
|
|
#endif
|
|
|
|
#define IS_SERVER_GADGET(id) \
|
|
((id) >= XW_SOLO_GADGET_ID && (id) <= XW_CLIENT_GADGET_ID)
|
|
|
|
|
|
/*****************************************************************************
|
|
*
|
|
****************************************************************************/
|
|
Boolean
|
|
newGameHandleEvent( EventPtr event )
|
|
{
|
|
Boolean result = false;
|
|
EventType eventToPost; /* used only with OK button */
|
|
PalmAppGlobals* globals;
|
|
FormPtr form;
|
|
CurGameInfo* gi;
|
|
PalmNewGameState* state;
|
|
Int16 chosen;
|
|
XP_U16 controlID;
|
|
Boolean on;
|
|
|
|
CALLBACK_PROLOGUE();
|
|
|
|
globals = getFormRefcon();
|
|
gi = &globals->gameInfo;
|
|
state = &globals->newGameState;
|
|
|
|
switch ( event->eType ) {
|
|
|
|
case frmOpenEvent:
|
|
|
|
GlobalPrefsToLocal( globals );
|
|
|
|
form = FrmGetActiveForm();
|
|
|
|
XP_ASSERT( !state->ngc );
|
|
XP_MEMSET( state, 0, sizeof(*state) );
|
|
|
|
state->form = form;
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
sizeGadgetsForStrings( form,
|
|
getActiveObjectPtr( XW_SERVERTYPES_LIST_ID ),
|
|
XW_SOLO_GADGET_ID );
|
|
#endif
|
|
loadNewGameState( globals );
|
|
if ( !globals->isNewGame ) {
|
|
modDisabledList( state, XW_DICT_SELECTOR_ID, NGEnableDisabled );
|
|
}
|
|
|
|
XP_ASSERT( !!state->dictName );
|
|
setNameThatFits( state );
|
|
|
|
XP_ASSERT( !!globals->game.server );
|
|
case frmUpdateEvent:
|
|
GrfSetState( false, false, false );
|
|
FrmDrawForm( FrmGetActiveForm() );
|
|
|
|
drawConnectGadgets( globals );
|
|
|
|
result = true;
|
|
break;
|
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_RELAY
|
|
case connsSettingChgEvent:
|
|
XP_ASSERT( globals->isNewGame );
|
|
state->connsSettingChanged = XP_TRUE;
|
|
break;
|
|
#endif
|
|
|
|
#ifdef HS_DUO_SUPPORT
|
|
case frmObjectFocusTakeEvent:
|
|
case frmObjectFocusLostEvent:
|
|
result = considerGadgetFocus( state, event );
|
|
break;
|
|
#endif
|
|
|
|
case penDownEvent:
|
|
result = checkHiliteGadget( globals, event, state );
|
|
break;
|
|
|
|
case keyDownEvent:
|
|
result = tryFieldNavigationKey( event->data.keyDown.chr )
|
|
|| tryDuoRockerKey( globals, event->data.keyDown.chr );
|
|
break;
|
|
|
|
case prefsChangedEvent:
|
|
state->forwardChange = true;
|
|
break;
|
|
|
|
case fldEnterEvent:
|
|
case ctlEnterEvent:
|
|
if ( onDisabledList( state, event->data.ctlEnter.controlID ) ) {
|
|
result = true;
|
|
beep();
|
|
}
|
|
break;
|
|
|
|
case ctlSelectEvent:
|
|
result = true;
|
|
controlID = event->data.ctlSelect.controlID;
|
|
on = event->data.ctlSelect.on;
|
|
switch ( controlID ) {
|
|
|
|
case XW_ROBOT_1_CHECKBOX_ID:
|
|
case XW_ROBOT_2_CHECKBOX_ID:
|
|
case XW_ROBOT_3_CHECKBOX_ID:
|
|
case XW_ROBOT_4_CHECKBOX_ID:
|
|
handleRobotChanged( state, controlID, on );
|
|
break;
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case XW_REMOTE_1_CHECKBOX_ID:
|
|
case XW_REMOTE_2_CHECKBOX_ID:
|
|
case XW_REMOTE_3_CHECKBOX_ID:
|
|
case XW_REMOTE_4_CHECKBOX_ID:
|
|
handleRemoteChanged( state, controlID, on );
|
|
break;
|
|
#endif
|
|
|
|
case XW_NPLAYERS_SELECTOR_ID:
|
|
XP_ASSERT( globals->isNewGame );
|
|
chosen = LstPopupList( state->playerNumList );
|
|
if ( chosen >= 0 ) {
|
|
NGValue value;
|
|
setSelectorFromList( XW_NPLAYERS_SELECTOR_ID,
|
|
state->playerNumList,
|
|
chosen );
|
|
value.ng_u16 = chosen + 1;
|
|
newg_attrChanged( state->ngc, NG_ATTR_NPLAYERS, value );
|
|
}
|
|
break;
|
|
|
|
case XW_DICT_SELECTOR_ID:
|
|
XP_ASSERT( globals->isNewGame );
|
|
globals->dictuiForBeaming = false;
|
|
FrmPopupForm( XW_DICTINFO_FORM );
|
|
|
|
/* popup dict selection dialog -- or maybe just a list if there
|
|
are no preferences to set. The results should all be
|
|
cancellable, so don't delete the existing dictionary (if
|
|
any) until OK is chosen */
|
|
break;
|
|
|
|
case XW_GINFO_JUGGLE_ID:
|
|
while ( !newg_juggle( state->ngc ) ) {
|
|
}
|
|
(void)newg_juggle( state->ngc );
|
|
break;
|
|
|
|
case XW_OK_BUTTON_ID:
|
|
|
|
/* if we put up the prefs form from within this one and the user
|
|
clicked ok, we need to make sure the main form gets the
|
|
notification so it can make use of any changes. This event
|
|
needs to arrive before the newGame event so any changes will
|
|
be incorporated. */
|
|
if ( state->forwardChange ) {
|
|
eventToPost.eType = prefsChangedEvent;
|
|
EvtAddEventToQueue( &eventToPost );
|
|
state->forwardChange = false;
|
|
}
|
|
|
|
updatePlayerInfo( globals );
|
|
if ( globals->isNewGame ) {
|
|
eventToPost.eType = newGameOkEvent;
|
|
EvtAddEventToQueue( &eventToPost );
|
|
globals->postponeDraw = true;
|
|
}
|
|
|
|
unloadNewGameState( globals );
|
|
|
|
FrmReturnToForm( 0 );
|
|
break;
|
|
|
|
case XW_CANCEL_BUTTON_ID:
|
|
unloadNewGameState( globals );
|
|
eventToPost.eType = newGameCancelEvent;
|
|
EvtAddEventToQueue( &eventToPost );
|
|
FrmReturnToForm( 0 );
|
|
break;
|
|
|
|
case XW_PREFS_BUTTON_ID:
|
|
/* bring up with the this-game tab selected */
|
|
XP_ASSERT( !!globals->prefsDlgState );
|
|
globals->prefsDlgState->stateTypeIsGlobal = false;
|
|
FrmPopupForm( XW_PREFS_FORM );
|
|
break;
|
|
|
|
case XW_PLAYERPASSWD_1_TRIGGER_ID:
|
|
case XW_PLAYERPASSWD_2_TRIGGER_ID:
|
|
case XW_PLAYERPASSWD_3_TRIGGER_ID:
|
|
case XW_PLAYERPASSWD_4_TRIGGER_ID:
|
|
handlePasswordTrigger( globals, controlID );
|
|
break;
|
|
|
|
default: /* one of the password selectors? */
|
|
result = false;
|
|
}
|
|
break;
|
|
|
|
case dictSelectedEvent:
|
|
/* posted by the form we raise when user clicks Dictionary selector
|
|
above. */
|
|
if ( state->dictName != NULL ) {
|
|
XP_FREE( globals->mpool, state->dictName );
|
|
}
|
|
state->dictName =
|
|
((DictSelectedData*)&event->data.generic)->dictName;
|
|
setNameThatFits( state );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
CALLBACK_EPILOGUE();
|
|
return result;
|
|
} /* newGameHandleEvent */
|
|
|
|
static void
|
|
setNameThatFits( PalmNewGameState* state )
|
|
{
|
|
RectangleType rect;
|
|
XP_U16 width;
|
|
XP_U16 len = XP_STRLEN( (const char*)state->dictName );
|
|
|
|
XP_MEMCPY( state->shortDictName, state->dictName, len + 1 );
|
|
|
|
/* The width available is the cancel button's left minus ours */
|
|
getObjectBounds( XW_CANCEL_BUTTON_ID, &rect );
|
|
width = rect.topLeft.x;
|
|
getObjectBounds( XW_DICT_SELECTOR_ID, &rect );
|
|
width -= (rect.topLeft.x + 6);
|
|
|
|
for ( ; FntCharsWidth( (const char*)state->dictName, len ) > width; --len ) {
|
|
/* do nothing */
|
|
}
|
|
|
|
state->shortDictName[len] = '\0';
|
|
CtlSetLabel( getActiveObjectPtr( XW_DICT_SELECTOR_ID ),
|
|
(const char*)state->shortDictName );
|
|
} /* setNameThatFits */
|
|
|
|
static Boolean
|
|
tryFieldNavigationKey( XP_U16 key )
|
|
{
|
|
FormPtr form;
|
|
Int16 curFocus, nextFocus, change;
|
|
UInt16 nObjects;
|
|
|
|
if ( key == prevFieldChr ) {
|
|
change = -1;
|
|
} else if ( key == nextFieldChr ) {
|
|
change = 1;
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
form = FrmGetActiveForm();
|
|
curFocus = nextFocus = FrmGetFocus( form );
|
|
nObjects = FrmGetNumberOfObjects(form);
|
|
|
|
/* find the next (in either direction) usable field */
|
|
for ( ; ; ) {
|
|
nextFocus += change;
|
|
|
|
if ( nextFocus == nObjects ) {
|
|
nextFocus = 0;
|
|
} else if ( nextFocus < 0 ) {
|
|
nextFocus = nObjects-1;
|
|
}
|
|
|
|
if ( nextFocus == curFocus ) {
|
|
break;
|
|
} else if ( FrmGetObjectType(form, nextFocus) != frmFieldObj ) {
|
|
continue;
|
|
} else {
|
|
FieldPtr field = FrmGetObjectPtr( form, nextFocus );
|
|
FieldAttrType attrs;
|
|
FldGetAttributes( field, &attrs );
|
|
if ( attrs.usable ) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
FrmSetFocus( form, nextFocus );
|
|
return true;
|
|
} /* tryFieldNavigationKey */
|
|
|
|
#ifdef HS_DUO_SUPPORT
|
|
#ifdef DEBUG
|
|
static XP_UCHAR*
|
|
keyToStr( XP_U16 key )
|
|
{
|
|
#define keyCase(k) case (k): return #k
|
|
switch( key ) {
|
|
keyCase(vchrRockerUp);
|
|
keyCase(vchrRockerDown);
|
|
keyCase(vchrRockerLeft);
|
|
keyCase(vchrRockerRight);
|
|
keyCase(vchrRockerCenter);
|
|
default:
|
|
return "huh?";
|
|
}
|
|
#undef keyCase
|
|
}
|
|
#endif
|
|
|
|
static XP_Bool
|
|
tryDuoRockerKey( PalmAppGlobals* globals, XP_U16 key )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
XP_U16 focusID;
|
|
FormPtr form;
|
|
|
|
switch( key ) {
|
|
case vchrRockerUp:
|
|
case vchrRockerDown:
|
|
case vchrRockerLeft:
|
|
case vchrRockerRight:
|
|
XP_LOGF( "got rocker key: %s", keyToStr(key) );
|
|
result = XP_FALSE;
|
|
break;
|
|
case vchrRockerCenter:
|
|
/* if one of the gadgets is focussed, "tap" it. */
|
|
XP_LOGF( "got rocker key: %s", keyToStr(key) );
|
|
form = FrmGetActiveForm();
|
|
focusID = FrmGetObjectId( form, FrmGetFocus( form ) );
|
|
if ( IS_SERVER_GADGET( focusID ) ) {
|
|
changeGadgetHilite( globals, focusID );
|
|
result = XP_TRUE;
|
|
} else {
|
|
XP_LOGF( "%d not server gadget", focusID );
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return result;
|
|
} /* tryDuoRockerKey */
|
|
#endif /* #ifdef HS_DUO_SUPPORT */
|
|
|
|
/*
|
|
* Copy the local state into global state.
|
|
*/
|
|
static void
|
|
updatePlayerInfo( PalmAppGlobals* globals )
|
|
{
|
|
CurGameInfo* gi;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
|
|
gi = &globals->gameInfo;
|
|
newg_store( state->ngc, gi );
|
|
|
|
gi->boardSize = globals->prefsDlgState->curBdSize;
|
|
|
|
replaceStringIfDifferent( globals->mpool, &gi->dictName,
|
|
globals->newGameState.dictName );
|
|
} /* updatePlayerInfo */
|
|
|
|
void
|
|
drawOneGadget( UInt16 id, char* text, Boolean hilite )
|
|
{
|
|
RectangleType divRect;
|
|
XP_U16 len = XP_STRLEN(text);
|
|
XP_U16 width = FntCharsWidth( text, len );
|
|
XP_U16 left;
|
|
|
|
getObjectBounds( id, &divRect );
|
|
WinDrawRectangleFrame( rectangleFrame, &divRect );
|
|
WinEraseRectangle( &divRect, 0 );
|
|
left = divRect.topLeft.x;
|
|
left += (divRect.extent.x - width) / 2;
|
|
WinDrawChars( text, len, left, divRect.topLeft.y );
|
|
if ( hilite ) {
|
|
WinInvertRectangle( &divRect, 0 );
|
|
}
|
|
} /* drawOneGadget */
|
|
|
|
/* Frame 'em, draw their text, and highlight the one that's selected
|
|
*/
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
|
|
#ifdef HS_DUO_SUPPORT
|
|
static void
|
|
drawFocusRingOnGadget()
|
|
{
|
|
FormPtr form = FrmGetActiveForm();
|
|
XP_U16 focusID = FrmGetObjectId( form, FrmGetFocus(form) );
|
|
if ( IS_SERVER_GADGET(focusID) ) {
|
|
Err err;
|
|
RectangleType rect;
|
|
|
|
getObjectBounds( focusID, &rect );
|
|
|
|
err = HsNavDrawFocusRing( form, focusID, 0, &rect,
|
|
hsNavFocusRingStyleObjectTypeDefault,
|
|
false );
|
|
XP_ASSERT( err == errNone );
|
|
}
|
|
} /* drawFocusRingOnGadget */
|
|
#endif
|
|
|
|
static void
|
|
drawConnectGadgets( PalmAppGlobals* globals )
|
|
{
|
|
UInt16 i;
|
|
ListPtr list = getActiveObjectPtr( XW_SERVERTYPES_LIST_ID );
|
|
XP_ASSERT( !!list );
|
|
|
|
for ( i = 0; i < 3; ++i ) {
|
|
char* text = LstGetSelectionText( list, i );
|
|
Boolean hilite = i == globals->newGameState.curServerHilite;
|
|
drawOneGadget( i + XW_SOLO_GADGET_ID, text, hilite );
|
|
}
|
|
|
|
#ifdef HS_DUO_SUPPORT
|
|
drawFocusRingOnGadget();
|
|
#endif
|
|
|
|
} /* drawConnectGadgets */
|
|
|
|
#ifdef HS_DUO_SUPPORT
|
|
static XP_Bool
|
|
considerGadgetFocus( PalmNewGameState* state, EventType* event )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
XP_Bool isTake;
|
|
XP_U16 eType = event->eType;
|
|
FormPtr form = FrmGetActiveForm();
|
|
XP_U16 objectID;
|
|
|
|
XP_ASSERT( eType == frmObjectFocusTakeEvent
|
|
|| eType == frmObjectFocusLostEvent );
|
|
XP_ASSERT( event->data.frmObjectFocusTake.formID == FrmGetActiveFormID() );
|
|
|
|
|
|
isTake = eType == frmObjectFocusTakeEvent;
|
|
if ( isTake ) {
|
|
objectID = event->data.frmObjectFocusTake.objectID;
|
|
} else {
|
|
objectID = event->data.frmObjectFocusLost.objectID;
|
|
}
|
|
|
|
/* docs say to return HANDLED for both take and lost */
|
|
result = IS_SERVER_GADGET( objectID );
|
|
|
|
if ( result ) {
|
|
Err err;
|
|
if ( isTake ) {
|
|
|
|
FrmSetFocus(form, FrmGetObjectIndex(form, objectID));
|
|
drawFocusRingOnGadget();
|
|
result = XP_TRUE;
|
|
/* } else { */
|
|
/* err = HsNavRemoveFocusRing( form ); */
|
|
}
|
|
}
|
|
|
|
return result;
|
|
} /* considerGadgetFocus */
|
|
#endif
|
|
|
|
static void
|
|
changeGadgetHilite( PalmAppGlobals* globals, UInt16 hiliteID )
|
|
{
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
XP_Bool isNewGame = globals->isNewGame;
|
|
|
|
hiliteID -= XW_SOLO_GADGET_ID;
|
|
|
|
if ( hiliteID != state->curServerHilite ) {
|
|
/* if it's not a new game, don't recognize the change */
|
|
if ( isNewGame ) {
|
|
NGValue value;
|
|
|
|
state->curServerHilite = hiliteID;
|
|
drawConnectGadgets( globals );
|
|
|
|
value.ng_role = hiliteID;
|
|
newg_attrChanged( state->ngc, NG_ATTR_ROLE, value );
|
|
} else {
|
|
beep();
|
|
}
|
|
}
|
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_RELAY
|
|
/* Even if it didn't change, pop the connections form. It's only
|
|
informational in the non-new-game case; nothing can be changed. */
|
|
if ( hiliteID != SERVER_STANDALONE ) {
|
|
if ( isNewGame || hiliteID==globals->newGameState.curServerHilite ) {
|
|
PopupConnsForm( globals, hiliteID, &state->addr );
|
|
}
|
|
}
|
|
#endif
|
|
} /* changeGadgetHilite */
|
|
|
|
static Boolean
|
|
checkHiliteGadget( PalmAppGlobals* globals, EventType* event,
|
|
PalmNewGameState* XP_UNUSED_DBG(state) )
|
|
{
|
|
Boolean result = false;
|
|
UInt16 selGadget;
|
|
|
|
XP_ASSERT( &globals->newGameState == state );
|
|
|
|
result = penInGadget( event, &selGadget );
|
|
if ( result ) {
|
|
changeGadgetHilite( globals, selGadget );
|
|
}
|
|
|
|
return result;
|
|
} /* checkHiliteGadget */
|
|
#endif
|
|
|
|
static void
|
|
showMaskedPwd( XP_U16 objectID, XP_Bool set )
|
|
{
|
|
const char* label = set? "*" : " ";
|
|
/* control owns the string passed in */
|
|
CtlSetLabel( getActiveObjectPtr( objectID ), label );
|
|
}
|
|
|
|
/* If there's currently no password set, just let 'em set one. If there is
|
|
* one set, they need to know the old before setting the new.
|
|
*/
|
|
static void
|
|
handlePasswordTrigger( PalmAppGlobals* globals, UInt16 controlID )
|
|
{
|
|
UInt16 playerNum;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
XP_UCHAR* name;
|
|
FieldPtr nameField;
|
|
XP_U16 len;
|
|
|
|
playerNum = (controlID - XW_PLAYERPASSWD_1_TRIGGER_ID) / NUM_PLAYER_COLS;
|
|
XP_ASSERT( playerNum < MAX_NUM_PLAYERS );
|
|
|
|
nameField = getActiveObjectPtr( XW_PLAYERNAME_1_FIELD_ID +
|
|
(NUM_PLAYER_COLS * playerNum) );
|
|
name = (XP_UCHAR*)FldGetTextPtr( nameField );
|
|
|
|
len = sizeof(state->passwds[playerNum]);
|
|
if ( askPassword( name, true, state->passwds[playerNum], &len )) {
|
|
showMaskedPwd( controlID, len > 0 );
|
|
}
|
|
} /* handlePasswordTrigger */
|
|
|
|
static void
|
|
unloadNewGameState( PalmAppGlobals* globals )
|
|
{
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
|
|
XP_FREE( globals->mpool, state->dictName );
|
|
state->dictName = NULL;
|
|
|
|
newg_destroy( state->ngc );
|
|
state->ngc = NULL;
|
|
} /* unloadNewGameState */
|
|
|
|
static XP_S16
|
|
findDisabled( const PalmNewGameState* state, XP_U16 controlID )
|
|
{
|
|
XP_U16 i;
|
|
XP_S16 loc = -1;
|
|
for ( i = 0; i < state->nDisabled; ++i ) {
|
|
if ( state->disabled[i] == controlID ) {
|
|
loc = i;
|
|
break;
|
|
}
|
|
}
|
|
return loc;
|
|
}
|
|
|
|
static XP_Bool
|
|
onDisabledList( const PalmNewGameState* state, XP_U16 controlID )
|
|
{
|
|
return 0 <= findDisabled( state, controlID );
|
|
}
|
|
|
|
static void
|
|
modDisabledList( PalmNewGameState* state, XP_U16 controlID,
|
|
NewGameEnable enable )
|
|
{
|
|
XP_S16 loc = findDisabled( state, controlID );
|
|
XP_Bool include = enable != NGEnableEnabled;
|
|
if ( loc < 0 && include ) {
|
|
/* not there but should be; add it */
|
|
state->disabled[state->nDisabled++] = controlID;
|
|
XP_ASSERT( state->nDisabled < MAX_DISABLED );
|
|
} else if ( loc >= 0 && !include ) {
|
|
/* is there; shouldn't be; remove it */
|
|
state->disabled[loc] = state->disabled[--state->nDisabled];
|
|
} else {
|
|
/* all's well */
|
|
XP_ASSERT( (loc >= 0 && include)
|
|
|| ( loc < 0 && !include ) );
|
|
}
|
|
}
|
|
|
|
static XP_U16
|
|
getBaseForCol( NewGameColumn col )
|
|
{
|
|
XP_U16 resID = 0;
|
|
switch ( col ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_COL_REMOTE:
|
|
resID = XW_REMOTE_1_CHECKBOX_ID;
|
|
break;
|
|
#endif
|
|
case NG_COL_NAME:
|
|
resID = XW_PLAYERNAME_1_FIELD_ID;
|
|
break;
|
|
case NG_COL_ROBOT:
|
|
resID = XW_ROBOT_1_CHECKBOX_ID;
|
|
break;
|
|
case NG_COL_PASSWD:
|
|
resID = XW_PLAYERPASSWD_1_TRIGGER_ID;
|
|
break;
|
|
default:
|
|
XP_ASSERT( XP_FALSE );
|
|
}
|
|
XP_ASSERT( !!resID );
|
|
return resID;
|
|
} /* getBaseForCol */
|
|
|
|
static XP_U16
|
|
objIDForCol( XP_U16 player, NewGameColumn col )
|
|
{
|
|
XP_U16 base = getBaseForCol( col );
|
|
return base + (NUM_PLAYER_COLS * player);
|
|
}
|
|
|
|
static ControlPtr
|
|
getControlForCol( XP_U16 player, NewGameColumn col )
|
|
{
|
|
XP_U16 objID = objIDForCol( player, col );
|
|
ControlPtr ctrl = getActiveObjectPtr( objID );
|
|
return ctrl;
|
|
}
|
|
|
|
static void
|
|
palmEnableColProc( void* closure, XP_U16 player, NewGameColumn col,
|
|
NewGameEnable enable )
|
|
{
|
|
PalmAppGlobals* globals = (PalmAppGlobals*)closure;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
XP_U16 objID = objIDForCol( player, col );
|
|
disOrEnable( state->form, objID, enable != NGEnableHidden );
|
|
modDisabledList( state, objID, enable );
|
|
}
|
|
|
|
/* Palm doesn't really do "disabled." Things are visible or not. But we
|
|
* want the player count dropdown in particular visible since it give
|
|
* information. So different objects get treated differently. The code
|
|
* handling ctlEnterEvent can disable for non-newgame dialogs even if a
|
|
* control is technically enabled.
|
|
*/
|
|
static void
|
|
palmEnableAttrProc(void* closure, NewGameAttr attr, NewGameEnable ngEnable )
|
|
{
|
|
PalmAppGlobals* globals = (PalmAppGlobals*)closure;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
XP_Bool enable = XP_FALSE; /* make compiler happy */
|
|
XP_U16 objID = 0;
|
|
|
|
switch ( attr ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_ATTR_ROLE:
|
|
/* always enabled */
|
|
break;
|
|
case NG_ATTR_REMHEADER:
|
|
enable = ngEnable != NGEnableHidden;
|
|
objID = XW_LOCAL_LABEL_ID;
|
|
break;
|
|
#endif
|
|
case NG_ATTR_NPLAYERS:
|
|
enable = ngEnable != NGEnableHidden;
|
|
objID = XW_NPLAYERS_SELECTOR_ID;
|
|
break;
|
|
case NG_ATTR_NPLAYHEADER:
|
|
break;
|
|
case NG_ATTR_CANJUGGLE:
|
|
enable = ngEnable == NGEnableEnabled;
|
|
objID = XW_GINFO_JUGGLE_ID;
|
|
break;
|
|
}
|
|
|
|
if ( objID != 0 ) {
|
|
disOrEnable( state->form, objID, enable );
|
|
modDisabledList( state, objID, ngEnable );
|
|
}
|
|
} /* palmEnableAttrProc */
|
|
|
|
static void
|
|
palmGetColProc( void* closure, XP_U16 player, NewGameColumn col,
|
|
NgCpCallbk cpcb, const void* cbClosure )
|
|
{
|
|
PalmAppGlobals* globals;
|
|
PalmNewGameState* state;
|
|
NGValue value;
|
|
XP_U16 objID = objIDForCol( player, col );
|
|
|
|
switch ( col ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_COL_REMOTE:
|
|
#endif
|
|
case NG_COL_ROBOT:
|
|
value.ng_bool = getBooleanCtrl( objID );
|
|
break;
|
|
case NG_COL_NAME:
|
|
value.ng_cp = FldGetTextPtr( getActiveObjectPtr( objID ) );
|
|
break;
|
|
case NG_COL_PASSWD:
|
|
globals = (PalmAppGlobals*)closure;
|
|
state = &globals->newGameState;
|
|
value.ng_cp = state->passwds[player];
|
|
break;
|
|
default:
|
|
XP_ASSERT(0);
|
|
}
|
|
|
|
(*cpcb)( value, cbClosure );
|
|
}
|
|
|
|
static void
|
|
palmSetColProc( void* closure, XP_U16 player, NewGameColumn col,
|
|
const NGValue value )
|
|
{
|
|
PalmAppGlobals* globals = (PalmAppGlobals*)closure;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
ControlPtr ctrl;
|
|
XP_U16 objID = objIDForCol( player, col );
|
|
|
|
switch ( col ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_COL_REMOTE:
|
|
#endif
|
|
case NG_COL_ROBOT:
|
|
ctrl = getControlForCol( player, col );
|
|
CtlSetValue( ctrl, value.ng_bool );
|
|
break;
|
|
case NG_COL_NAME:
|
|
setFieldStr( objID, value.ng_cp );
|
|
break;
|
|
case NG_COL_PASSWD:
|
|
if ( !!value.ng_cp ) {
|
|
XP_SNPRINTF( state->passwds[player], sizeof(state->passwds[player]),
|
|
"%s", value.ng_cp );
|
|
showMaskedPwd( objID, *value.ng_cp != '\0' );
|
|
}
|
|
|
|
default: /* shut up compiler */
|
|
break;
|
|
}
|
|
} /* palmSetColProc */
|
|
|
|
static void
|
|
palmSetAttrProc( void* closure, NewGameAttr attr, const NGValue value )
|
|
{
|
|
PalmAppGlobals* globals = (PalmAppGlobals*)closure;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
|
|
switch ( attr ) {
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case NG_ATTR_ROLE:
|
|
state->curServerHilite = value.ng_role;
|
|
/* Don't do this until frmUpdateEvent's been received... */
|
|
/* drawConnectGadgets( globals ); */
|
|
break;
|
|
case NG_ATTR_REMHEADER:
|
|
break;
|
|
#endif
|
|
case NG_ATTR_NPLAYERS:
|
|
setSelectorFromList( XW_NPLAYERS_SELECTOR_ID,
|
|
state->playerNumList, value.ng_u16 - 1 );
|
|
break;
|
|
case NG_ATTR_NPLAYHEADER: {
|
|
FieldPtr field = getActiveObjectPtr( XW_TOTALP_FIELD_ID );
|
|
char* cur = FldGetTextPtr( field );
|
|
/* Need to update to get it drawn and that flashes the whole dialog.
|
|
So avoid if possible. */
|
|
if ( !cur || (0 != StrCompare( cur, value.ng_cp ) ) ) {
|
|
FldSetTextPtr( field, (char*)value.ng_cp );
|
|
FrmUpdateForm( 0, frmRedrawUpdateCode );
|
|
}
|
|
}
|
|
break;
|
|
case NG_ATTR_CANJUGGLE:
|
|
XP_ASSERT(0); /* doesn't make sense */
|
|
break;
|
|
}
|
|
} /* palmSetAttrProc */
|
|
|
|
static XP_U16
|
|
palmPlayerFromID( XP_U16 id, XP_U16 base )
|
|
{
|
|
XP_U16 player;
|
|
player = (id - base) / NUM_PLAYER_COLS;
|
|
return player;
|
|
} /* palmPlayerFromID */
|
|
|
|
static void
|
|
handleRobotChanged( PalmNewGameState* state, XP_U16 controlID, XP_Bool on )
|
|
{
|
|
XP_U16 player;
|
|
NGValue value;
|
|
|
|
player = palmPlayerFromID( controlID, XW_ROBOT_1_CHECKBOX_ID );
|
|
value.ng_bool = on;
|
|
newg_colChanged( state->ngc, player );
|
|
}
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
static void
|
|
handleRemoteChanged( PalmNewGameState* state, XP_U16 controlID, XP_Bool on )
|
|
{
|
|
XP_U16 player;
|
|
NGValue value;
|
|
|
|
XP_LOGF( "%s: controlID=%d", __FUNCTION__, controlID );
|
|
|
|
player = palmPlayerFromID( controlID, XW_REMOTE_1_CHECKBOX_ID );
|
|
value.ng_bool = on;
|
|
newg_colChanged( state->ngc, player );
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
loadNewGameState( PalmAppGlobals* globals )
|
|
{
|
|
CurGameInfo* gi = &globals->gameInfo;
|
|
PalmNewGameState* state = &globals->newGameState;
|
|
|
|
state->dictName = copyString( globals->mpool, gi->dictName );
|
|
state->playerNumList = getActiveObjectPtr( XW_NPLAYERS_LIST_ID );
|
|
|
|
state->ngc = newg_make( MPPARM(globals->mpool)
|
|
globals->isNewGame,
|
|
&globals->util,
|
|
palmEnableColProc,
|
|
palmEnableAttrProc,
|
|
palmGetColProc,
|
|
palmSetColProc,
|
|
palmSetAttrProc,
|
|
globals );
|
|
newg_load( state->ngc, gi );
|
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_RELAY
|
|
if ( globals->game.comms ) {
|
|
comms_getAddr( globals->game.comms, &state->addr );
|
|
} else {
|
|
comms_getInitialAddr( &state->addr );
|
|
}
|
|
#endif
|
|
} /* loadNewGameState */
|