mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-07 05:24:46 +01:00
new game works
Only two options: local game with one robot, or networked needing to invite someone.
This commit is contained in:
parent
57783756d6
commit
43874a6a5d
5 changed files with 162 additions and 57 deletions
|
@ -72,7 +72,7 @@ main.html: ${INPUTS} Makefile shell_minimal.html
|
||||||
--preload-file assets_dir --shell-file shell_minimal.html \
|
--preload-file assets_dir --shell-file shell_minimal.html \
|
||||||
-s USE_SDL_TTF=2 -s USE_SDL_IMAGE=2 -s SDL2_IMAGE_FORMATS='["png"]' \
|
-s USE_SDL_TTF=2 -s USE_SDL_IMAGE=2 -s SDL2_IMAGE_FORMATS='["png"]' \
|
||||||
-s "EXTRA_EXPORTED_RUNTIME_METHODS=['ccall']" \
|
-s "EXTRA_EXPORTED_RUNTIME_METHODS=['ccall']" \
|
||||||
-s EXPORTED_FUNCTIONS='["_main", "_gotMQTTMsg", "_jscallback", "_onDlgButton", "_MQTTConnectedChanged"]' \
|
-s EXPORTED_FUNCTIONS='["_main", "_gotMQTTMsg", "_jscallback", "_onDlgButton", "_MQTTConnectedChanged", "_onNewGame"]' \
|
||||||
-s WASM=1 \
|
-s WASM=1 \
|
||||||
${INPUTS} -o $@
|
${INPUTS} -o $@
|
||||||
|
|
||||||
|
|
|
@ -83,9 +83,13 @@
|
||||||
#define BUTTON_GAME_DELETE "Delete Game"
|
#define BUTTON_GAME_DELETE "Delete Game"
|
||||||
#define MAX_BUTTONS 20 /* not sure what's safe here */
|
#define MAX_BUTTONS 20 /* not sure what's safe here */
|
||||||
|
|
||||||
|
typedef struct _NewGameParams {
|
||||||
|
bool isRobotNotRemote;
|
||||||
|
bool hintsNotAllowed;
|
||||||
|
} NewGameParams;
|
||||||
|
|
||||||
static void loadAndDraw( Globals* globals, const NetLaunchInfo* invite,
|
static void loadAndDraw( Globals* globals, const NetLaunchInfo* invite,
|
||||||
const char* key, bool forceNew );
|
const char* key, NewGameParams* params );
|
||||||
static void nameGame( Globals* globals );
|
static void nameGame( Globals* globals );
|
||||||
static void ensureName( Globals* globals );
|
static void ensureName( Globals* globals );
|
||||||
static void loadName( Globals* globals );
|
static void loadName( Globals* globals );
|
||||||
|
@ -205,6 +209,11 @@ EM_JS(void, setButtons, (const char* id, const char** bstrs,
|
||||||
setDivButtons(UTF8ToString(id), buttons, proc, closure);
|
setDivButtons(UTF8ToString(id), buttons, proc, closure);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
EM_JS(void, callNewGame, (const char* msg, void* closure), {
|
||||||
|
let jsmsg = UTF8ToString(msg);
|
||||||
|
nbGetNewGame(closure, jsmsg);
|
||||||
|
});
|
||||||
|
|
||||||
static void updateScreen( Globals* globals, bool doSave );
|
static void updateScreen( Globals* globals, bool doSave );
|
||||||
|
|
||||||
typedef void (*ConfirmProc)( void* closure, bool confirmed );
|
typedef void (*ConfirmProc)( void* closure, bool confirmed );
|
||||||
|
@ -290,11 +299,17 @@ formatGameID( char* buf, size_t len, int gameID )
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
formatGameKey( char* buf, size_t len, const char* gameID )
|
formatGameKeyStr( char* buf, size_t len, const char* gameID )
|
||||||
{
|
{
|
||||||
snprintf( buf, len, KEY_GAME_PREFIX "%s", gameID );
|
snprintf( buf, len, KEY_GAME_PREFIX "%s", gameID );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
formatGameKeyInt( char* buf, size_t len, int gameID )
|
||||||
|
{
|
||||||
|
snprintf( buf, len, KEY_GAME_PREFIX "%X", gameID );
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
formatNameKey( char* buf, size_t len, int gameID )
|
formatNameKey( char* buf, size_t len, int gameID )
|
||||||
{
|
{
|
||||||
|
@ -342,12 +357,13 @@ onGameButton( void* closure, const char* button )
|
||||||
static void
|
static void
|
||||||
updateGameButtons( Globals* globals )
|
updateGameButtons( Globals* globals )
|
||||||
{
|
{
|
||||||
GameStateInfo gsi;
|
|
||||||
game_getState( &globals->gs.game, NULL, &gsi );
|
|
||||||
|
|
||||||
const char* buttons[MAX_BUTTONS];
|
const char* buttons[MAX_BUTTONS];
|
||||||
int cur = 0;
|
int cur = 0;
|
||||||
|
|
||||||
|
if ( !!globals->gs.util ) {
|
||||||
|
GameStateInfo gsi;
|
||||||
|
game_getState( &globals->gs.game, NULL, &gsi );
|
||||||
|
|
||||||
if ( gsi.canHint ) {
|
if ( gsi.canHint ) {
|
||||||
buttons[cur++] = BUTTON_HINTDOWN;
|
buttons[cur++] = BUTTON_HINTDOWN;
|
||||||
buttons[cur++] = BUTTON_HINTUP;
|
buttons[cur++] = BUTTON_HINTUP;
|
||||||
|
@ -368,7 +384,7 @@ updateGameButtons( Globals* globals )
|
||||||
}
|
}
|
||||||
|
|
||||||
buttons[cur++] = BUTTON_VALS;
|
buttons[cur++] = BUTTON_VALS;
|
||||||
|
}
|
||||||
buttons[cur++] = NULL;
|
buttons[cur++] = NULL;
|
||||||
|
|
||||||
setButtons( BUTTONS_ID_GAME, buttons, onGameButton, globals );
|
setButtons( BUTTONS_ID_GAME, buttons, onGameButton, globals );
|
||||||
|
@ -378,7 +394,7 @@ static void
|
||||||
onGameChosen( void* closure, const char* key )
|
onGameChosen( void* closure, const char* key )
|
||||||
{
|
{
|
||||||
Globals* globals = (Globals*)closure;
|
Globals* globals = (Globals*)closure;
|
||||||
loadAndDraw( globals, NULL, key, false );
|
loadAndDraw( globals, NULL, key, NULL );
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -394,13 +410,47 @@ onGameRanamed( void* closure, const char* newName )
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
cleanupGame( Globals* globals )
|
||||||
|
{
|
||||||
|
if ( !!globals->gs.util ) {
|
||||||
|
game_dispose( &globals->gs.game, NULL );
|
||||||
|
wasm_util_destroy( globals->gs.util );
|
||||||
|
XP_MEMSET( &globals->gs, 0, sizeof(globals->gs) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
deleteCurGame( Globals* globals )
|
||||||
|
{
|
||||||
|
int gameID = globals->gs.gi.gameID; /* remember it */
|
||||||
|
cleanupGame( globals );
|
||||||
|
|
||||||
|
char key[32];
|
||||||
|
formatNameKey( key, sizeof(key), gameID );
|
||||||
|
remove_stored_value( key );
|
||||||
|
|
||||||
|
formatGameKeyInt( key, sizeof(key), gameID );
|
||||||
|
remove_stored_value( key );
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
onDeleteConfirmed( void* closure, bool confirmed )
|
||||||
|
{
|
||||||
|
if ( confirmed ) {
|
||||||
|
Globals* globals = (Globals*)closure;
|
||||||
|
deleteCurGame( globals );
|
||||||
|
updateScreen( globals, false );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
onDeviceButton( void* closure, const char* button )
|
onDeviceButton( void* closure, const char* button )
|
||||||
{
|
{
|
||||||
Globals* globals = (Globals*)closure;
|
Globals* globals = (Globals*)closure;
|
||||||
XP_LOGFF( "(button=%s)", button );
|
XP_LOGFF( "(button=%s)", button );
|
||||||
if ( 0 == strcmp(button, BUTTON_GAME_NEW) ) {
|
if ( 0 == strcmp(button, BUTTON_GAME_NEW) ) {
|
||||||
loadAndDraw( globals, NULL, NULL, true );
|
callNewGame("Configure your new game", globals);
|
||||||
} else if ( 0 == strcmp(button, BUTTON_GAME_OPEN) ) {
|
} else if ( 0 == strcmp(button, BUTTON_GAME_OPEN) ) {
|
||||||
const char* msg = "Choose game to open";
|
const char* msg = "Choose game to open";
|
||||||
call_pickGame( msg, onGameChosen, globals);
|
call_pickGame( msg, onGameChosen, globals);
|
||||||
|
@ -409,6 +459,11 @@ onDeviceButton( void* closure, const char* button )
|
||||||
call_get_string( "Rename your game", globals->gs.gameName,
|
call_get_string( "Rename your game", globals->gs.gameName,
|
||||||
onGameRanamed, globals );
|
onGameRanamed, globals );
|
||||||
} else if ( 0 == strcmp(button, BUTTON_GAME_DELETE) ) {
|
} else if ( 0 == strcmp(button, BUTTON_GAME_DELETE) ) {
|
||||||
|
char msg[256];
|
||||||
|
snprintf( msg, sizeof(msg), "Are you sure you want to delete the game \"%s\"?"
|
||||||
|
"\nThis action cannot be undone.",
|
||||||
|
globals->gs.gameName );
|
||||||
|
call_confirm( globals, msg, onDeleteConfirmed, globals );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -492,16 +547,6 @@ startGame( Globals* globals, const char* name )
|
||||||
LOG_RETURN_VOID();
|
LOG_RETURN_VOID();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
cleanupGame( Globals* globals )
|
|
||||||
{
|
|
||||||
if ( !!globals->gs.util ) {
|
|
||||||
game_dispose( &globals->gs.game, NULL );
|
|
||||||
wasm_util_destroy( globals->gs.util );
|
|
||||||
XP_MEMSET( &globals->gs, 0, sizeof(globals->gs) );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef struct _AskReplaceState {
|
typedef struct _AskReplaceState {
|
||||||
Globals* globals;
|
Globals* globals;
|
||||||
NetLaunchInfo invite;
|
NetLaunchInfo invite;
|
||||||
|
@ -588,7 +633,7 @@ loadSavedGame( Globals* globals, const char* key )
|
||||||
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(globals->mpool)
|
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(globals->mpool)
|
||||||
globals->vtMgr );
|
globals->vtMgr );
|
||||||
char buf[32];
|
char buf[32];
|
||||||
formatGameKey( buf, sizeof(buf), key );
|
formatGameKeyStr( buf, sizeof(buf), key );
|
||||||
dutil_loadStream( globals->dutil, NULL, buf, NULL, stream );
|
dutil_loadStream( globals->dutil, NULL, buf, NULL, stream );
|
||||||
if ( 0 < stream_getSize( stream ) ) {
|
if ( 0 < stream_getSize( stream ) ) {
|
||||||
XP_ASSERT( !globals->gs.util );
|
XP_ASSERT( !globals->gs.util );
|
||||||
|
@ -653,12 +698,12 @@ nameGame( Globals* globals )
|
||||||
|
|
||||||
static void
|
static void
|
||||||
loadAndDraw( Globals* globals, const NetLaunchInfo* invite,
|
loadAndDraw( Globals* globals, const NetLaunchInfo* invite,
|
||||||
const char* key, bool forceNew )
|
const char* key, NewGameParams* params )
|
||||||
{
|
{
|
||||||
cleanupGame( globals );
|
cleanupGame( globals );
|
||||||
|
|
||||||
bool haveGame;
|
bool haveGame;
|
||||||
if ( forceNew ) {
|
if ( !!params ) {
|
||||||
haveGame = false;
|
haveGame = false;
|
||||||
} else {
|
} else {
|
||||||
/* First, load any saved game. We need it e.g. to confirm that an incoming
|
/* First, load any saved game. We need it e.g. to confirm that an incoming
|
||||||
|
@ -672,21 +717,22 @@ loadAndDraw( Globals* globals, const NetLaunchInfo* invite,
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( !haveGame ) {
|
if ( !haveGame ) {
|
||||||
bool p0robot = false;
|
globals->gs.gi.serverRole = !!params && !params->isRobotNotRemote
|
||||||
bool p1robot = true;
|
? SERVER_ISSERVER : SERVER_STANDALONE;
|
||||||
globals->gs.gi.serverRole = SERVER_STANDALONE;
|
|
||||||
globals->gs.gi.phoniesAction = PHONIES_WARN;
|
globals->gs.gi.phoniesAction = PHONIES_WARN;
|
||||||
globals->gs.gi.hintsNotAllowed = false;
|
globals->gs.gi.hintsNotAllowed = !!params && params->hintsNotAllowed || false;
|
||||||
globals->gs.gi.gameID = 0;
|
globals->gs.gi.gameID = 0;
|
||||||
globals->gs.gi.nPlayers = 2;
|
globals->gs.gi.nPlayers = 2;
|
||||||
globals->gs.gi.boardSize = 15;
|
globals->gs.gi.boardSize = 15;
|
||||||
globals->gs.gi.players[0].name = copyString( globals->mpool, "Player 1" );
|
globals->gs.gi.players[0].name = copyString( globals->mpool, "Player 1" ); /* FIXME */
|
||||||
globals->gs.gi.players[0].isLocal = XP_TRUE;
|
globals->gs.gi.players[0].isLocal = XP_TRUE;
|
||||||
globals->gs.gi.players[0].robotIQ = p0robot ? 99 : 0;
|
globals->gs.gi.players[0].robotIQ = 0;
|
||||||
|
|
||||||
globals->gs.gi.players[1].name = copyString( globals->mpool, "Player 2" );
|
globals->gs.gi.players[1].name = copyString( globals->mpool, "Player 2" );
|
||||||
globals->gs.gi.players[1].isLocal = XP_TRUE;
|
globals->gs.gi.players[1].isLocal = !!params ? params->isRobotNotRemote : true;
|
||||||
globals->gs.gi.players[1].robotIQ = p1robot ? 99 : 0;
|
XP_LOGFF( "set isLocal[1]: %d", globals->gs.gi.players[1].isLocal );
|
||||||
|
globals->gs.gi.players[1].robotIQ = 99; /* doesn't matter if remote */
|
||||||
|
|
||||||
globals->gs.util = wasm_util_make( MPPARM(globals->mpool) &globals->gs.gi,
|
globals->gs.util = wasm_util_make( MPPARM(globals->mpool) &globals->gs.gi,
|
||||||
globals->dutil, globals );
|
globals->dutil, globals );
|
||||||
|
@ -1001,8 +1047,10 @@ static void
|
||||||
updateScreen( Globals* globals, bool doSave )
|
updateScreen( Globals* globals, bool doSave )
|
||||||
{
|
{
|
||||||
SDL_RenderClear( globals->renderer );
|
SDL_RenderClear( globals->renderer );
|
||||||
|
if ( !!globals->gs.game.board ) {
|
||||||
board_draw( globals->gs.game.board, NULL );
|
board_draw( globals->gs.game.board, NULL );
|
||||||
wasm_draw_render( globals->draw, globals->renderer );
|
wasm_draw_render( globals->draw, globals->renderer );
|
||||||
|
}
|
||||||
SDL_RenderPresent( globals->renderer );
|
SDL_RenderPresent( globals->renderer );
|
||||||
|
|
||||||
updateGameButtons( globals );
|
updateGameButtons( globals );
|
||||||
|
@ -1014,14 +1062,14 @@ updateScreen( Globals* globals, bool doSave )
|
||||||
game_saveToStream( &globals->gs.game, NULL, &globals->gs.gi,
|
game_saveToStream( &globals->gs.game, NULL, &globals->gs.gi,
|
||||||
stream, ++globals->gs.saveToken );
|
stream, ++globals->gs.saveToken );
|
||||||
|
|
||||||
char gidBuf[16];
|
|
||||||
formatGameID( gidBuf, sizeof(gidBuf), globals->gs.gi.gameID );
|
|
||||||
char buf[32];
|
char buf[32];
|
||||||
formatGameKey( buf, sizeof(buf), gidBuf );
|
formatGameKeyInt( buf, sizeof(buf), globals->gs.gi.gameID );
|
||||||
dutil_storeStream( globals->dutil, NULL, buf, stream );
|
dutil_storeStream( globals->dutil, NULL, buf, stream );
|
||||||
stream_destroy( stream, NULL );
|
stream_destroy( stream, NULL );
|
||||||
game_saveSucceeded( &globals->gs.game, NULL, globals->gs.saveToken );
|
game_saveSucceeded( &globals->gs.game, NULL, globals->gs.saveToken );
|
||||||
|
|
||||||
|
char gidBuf[16];
|
||||||
|
formatGameID( gidBuf, sizeof(gidBuf), globals->gs.gi.gameID );
|
||||||
set_stored_value( KEY_LAST, gidBuf );
|
set_stored_value( KEY_LAST, gidBuf );
|
||||||
XP_LOGFF( "saved KEY_LAST: %s", gidBuf );
|
XP_LOGFF( "saved KEY_LAST: %s", gidBuf );
|
||||||
}
|
}
|
||||||
|
@ -1135,7 +1183,7 @@ initNoReturn( int argc, const char** argv )
|
||||||
|
|
||||||
const char* lastKey = get_stored_value( KEY_LAST );
|
const char* lastKey = get_stored_value( KEY_LAST );
|
||||||
XP_LOGFF( "loaded KEY_LAST: %s", lastKey );
|
XP_LOGFF( "loaded KEY_LAST: %s", lastKey );
|
||||||
loadAndDraw( globals, nlip, lastKey, false );
|
loadAndDraw( globals, nlip, lastKey, NULL );
|
||||||
if ( !!lastKey ) {
|
if ( !!lastKey ) {
|
||||||
free( (void*)lastKey );
|
free( (void*)lastKey );
|
||||||
}
|
}
|
||||||
|
@ -1177,6 +1225,17 @@ onDlgButton( AlertProc proc, void* closure, const char* button )
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
onNewGame( void* closure, bool opponentIsRobot )
|
||||||
|
{
|
||||||
|
Globals* globals = (Globals*)closure;
|
||||||
|
XP_LOGFF( "isRobot: %d", opponentIsRobot );
|
||||||
|
|
||||||
|
NewGameParams ngp = {0};
|
||||||
|
ngp.isRobotNotRemote = opponentIsRobot;
|
||||||
|
loadAndDraw( globals, NULL, NULL, &ngp );
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
main( int argc, const char** argv )
|
main( int argc, const char** argv )
|
||||||
{
|
{
|
||||||
|
|
|
@ -38,3 +38,8 @@ EM_JS(void, set_stored_value, (const char* key, const char* val), {
|
||||||
var jsVal = UTF8ToString(val);
|
var jsVal = UTF8ToString(val);
|
||||||
var jsString = localStorage.setItem(jsKey, jsVal);
|
var jsString = localStorage.setItem(jsKey, jsVal);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
EM_JS(void, remove_stored_value, (const char* key), {
|
||||||
|
var jsKey = UTF8ToString(key);
|
||||||
|
var jsString = localStorage.removeItem(jsKey);
|
||||||
|
});
|
||||||
|
|
|
@ -22,5 +22,6 @@
|
||||||
|
|
||||||
const char* get_stored_value( const char* key );
|
const char* get_stored_value( const char* key );
|
||||||
void set_stored_value(const char* key, const char* val);
|
void set_stored_value(const char* key, const char* val);
|
||||||
|
void remove_stored_value(const char* key);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -200,8 +200,48 @@ function nbGetString(msg, dflt, proc, closure) {
|
||||||
dlg.appendChild( newButtonDiv( buttons, butProc ) );
|
dlg.appendChild( newButtonDiv( buttons, butProc ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function newRadio(txt, id, proc) {
|
||||||
|
let span = document.createElement('span');
|
||||||
|
let radio = document.createElement('input');
|
||||||
|
radio.type = 'radio';
|
||||||
|
radio.id = id;
|
||||||
|
radio.name = id;
|
||||||
|
radio.onclick = proc;
|
||||||
|
|
||||||
|
let label = document.createElement('label')
|
||||||
|
label.htmlFor = id;
|
||||||
|
var description = document.createTextNode(txt);
|
||||||
|
label.appendChild(description);
|
||||||
|
|
||||||
|
span.appendChild(label);
|
||||||
|
span.appendChild(radio);
|
||||||
|
|
||||||
|
return span;
|
||||||
|
}
|
||||||
|
|
||||||
|
function nbGetNewGame(closure, msg) {
|
||||||
|
let dlg = newDlgWMsg('Is your opponent a robot or someone you will invite?');
|
||||||
|
|
||||||
|
let radioDiv = document.createElement('div');
|
||||||
|
dlg.appendChild( radioDiv );
|
||||||
|
var robotSet = [null];
|
||||||
|
radioDiv.appendChild(newRadio('Robot', 'newgame', function() {robotSet[0] = true;}));
|
||||||
|
radioDiv.appendChild(newRadio('Remote player', 'newgame', function() {robotSet[0] = false;}));
|
||||||
|
|
||||||
|
butProc = function(str) {
|
||||||
|
if ( str === 'OK' && null !== robotSet[0]) {
|
||||||
|
types = ['number', 'boolean'];
|
||||||
|
params = [closure, robotSet[0]];
|
||||||
|
Module.ccall('onNewGame', null, types, params);
|
||||||
|
}
|
||||||
|
dlg.parentNode.removeChild(dlg);
|
||||||
|
}
|
||||||
|
|
||||||
|
dlg.appendChild( newButtonDiv( ['Cancel', 'OK'], butProc ) );
|
||||||
|
}
|
||||||
|
|
||||||
for ( let one of ['paho-mqtt.js'] ) {
|
for ( let one of ['paho-mqtt.js'] ) {
|
||||||
let script = document.createElement('script');
|
let script = document.createElement('script');
|
||||||
script.src = one
|
script.src = one;
|
||||||
document.body.appendChild(script);
|
document.body.appendChild(script);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue