From 1b6ce4861f0d2326416bfa7315a1832d2bced85e Mon Sep 17 00:00:00 2001 From: Eric House Date: Sun, 28 Jun 2015 09:00:41 -0700 Subject: [PATCH 001/224] don't crash when invitation dialog cancelled --- xwords4/linux/gtkaskm.c | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/xwords4/linux/gtkaskm.c b/xwords4/linux/gtkaskm.c index c251eb8a0..aef7ef644 100644 --- a/xwords4/linux/gtkaskm.c +++ b/xwords4/linux/gtkaskm.c @@ -74,12 +74,13 @@ gtkaskm( const gchar* message, AskMInfo* infos, int nInfos ) gtk_main(); - if ( !state.cancelled ) { - for ( ii = 0; ii < nInfos; ++ii ) { + for ( ii = 0; ii < nInfos; ++ii ) { + AskMInfo* info = &infos[ii]; + if ( !state.cancelled ) { const gchar* txt = gtk_entry_get_text( GTK_ENTRY(fields[ii]) ); XP_LOGF( "%s: got text %s", __func__, txt ); - AskMInfo* info = &infos[ii]; - *info->result = g_strdup( txt ); + } else { + *info->result = NULL; } } From 0102cde2c3c5112de87673aebedc48a632ae05eb Mon Sep 17 00:00:00 2001 From: Eric House Date: Wed, 1 Jul 2015 06:34:57 -0700 Subject: [PATCH 002/224] add and handle new INVITE message types. (This can safely be added to shipping relay.) --- xwords4/relay/xwrelay.cpp | 42 +++++++++++++++++++++++++++++++++++---- xwords4/relay/xwrelay.h | 5 +++++ 2 files changed, 43 insertions(+), 4 deletions(-) diff --git a/xwords4/relay/xwrelay.cpp b/xwords4/relay/xwrelay.cpp index 1f155a2c8..8ea8c03cd 100644 --- a/xwords4/relay/xwrelay.cpp +++ b/xwords4/relay/xwrelay.cpp @@ -374,10 +374,15 @@ getHeader( const uint8_t** bufpp, const uint8_t* end, uint8_t byt; if ( getNetByte( bufpp, end, &byt ) ) { header->proto = (XWPDevProto)byt; - if ( XWPDEV_PROTO_VERSION_1 == header->proto - && vli2un( bufpp, end, &header->packetID ) - && getNetByte( bufpp, end, &byt ) - && byt < XWPDEV_N_ELEMS ) { + if ( XWPDEV_PROTO_VERSION_1 != header->proto ) { + logf( XW_LOGERROR, "%s: bad proto %d", __func__, header->proto ); + } else if ( !vli2un( bufpp, end, &header->packetID ) ) { + logf( XW_LOGERROR, "%s: can't get packet id", __func__ ); + } else if ( !getNetByte( bufpp, end, &byt ) ) { + logf( XW_LOGERROR, "%s: can't get cmd", __func__ ); + } else if ( XWPDEV_N_ELEMS <= byt ) { + logf( XW_LOGERROR, "%s: cmd %d too high", __func__, byt ); + } else { header->cmd = (XWRelayReg)byt; success = true; } @@ -838,6 +843,22 @@ post_upgrade( DevIDRelay devid ) (void)post_or_store( devid, packet, packetID, NULL, NULL ); } +void +post_invite( DevIDRelay sender, DevIDRelay invitee, const uint8_t* ptr, size_t len ) +{ + vector packet; + uint32_t packetID; + sender = htonl( sender ); + assemble_packet( packet, &packetID, XWPDEV_GOTINVITE, + &sender, sizeof(sender), + ptr, len, + NULL ); + + bool sent = post_or_store( invitee, packet, packetID, NULL, NULL ); + logf( XW_LOGINFO, "%s(): post_or_store => %s", __func__, + sent ? "sent" : "stored"); +} + /* A CONNECT message from a device gives us the hostID and socket we'll * associate with one participant in a relayed session. We'll store this * information with the cookie where other participants can find it when they @@ -1652,6 +1673,7 @@ msgToStr( XWRelayReg msg ) CASE_STR(XWPDEV_UNAVAIL); CASE_STR(XWPDEV_REG); CASE_STR(XWPDEV_REGRSP); + CASE_STR(XWPDEV_INVITE); CASE_STR(XWPDEV_KEEPALIVE); CASE_STR(XWPDEV_HAVEMSGS); CASE_STR(XWPDEV_RQSTMSGS); @@ -1756,6 +1778,18 @@ handle_udp_packet( UdpThreadClosure* utc ) } break; } + + case XWPDEV_INVITE: + DevIDRelay sender; + DevIDRelay invitee; + if ( getNetLong( &ptr, end, &sender ) + && getNetLong( &ptr, end, &invitee) ) { + logf( XW_LOGVERBOSE0, "got invite from %d for %d", + sender, invitee ); + post_invite( sender, invitee, ptr, end - ptr ); + } + break; + case XWPDEV_KEEPALIVE: case XWPDEV_RQSTMSGS: { DevID devID( ID_TYPE_RELAY ); diff --git a/xwords4/relay/xwrelay.h b/xwords4/relay/xwrelay.h index 2eef03edf..a502957c0 100644 --- a/xwords4/relay/xwrelay.h +++ b/xwords4/relay/xwrelay.h @@ -108,6 +108,11 @@ enum { XWPDEV_NONE /* 0 is an illegal value */ to check for upgrades; may eventually replace device needing a timer. */ + ,XWPDEV_INVITE /* dev->relay; format: header, sender relay + id: 4, recipient relay id: 4; nli data: + variable length. */ + ,XWPDEV_GOTINVITE /* relay->dev */ + ,XWPDEV_N_ELEMS /* MUST BE LAST */ } From 6d768972db67b7e004e115cb0fde3c53e6c6b51d Mon Sep 17 00:00:00 2001 From: Eric House Date: Wed, 1 Jul 2015 06:35:47 -0700 Subject: [PATCH 003/224] fix compile warning. (Not really tested...it's a test app) --- xwords4/relay/rq.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/xwords4/relay/rq.c b/xwords4/relay/rq.c index 6f7ddf8c0..4abe52808 100644 --- a/xwords4/relay/rq.c +++ b/xwords4/relay/rq.c @@ -252,8 +252,8 @@ connect_socket( void ) struct hostent* hostip; hostip = gethostbyname( g_host ); - memcpy( &(to_sock.sin_addr.s_addr), hostip->h_addr_list[0], - sizeof(hostip->h_addr_list[0] ) ); + memcpy( &(to_sock.sin_addr.s_addr), &hostip->h_addr_list[0], + sizeof(hostip->h_addr_list[0]) ); if ( 0 != connect( sockfd, (const struct sockaddr*)&to_sock, sizeof(to_sock) ) ) { From 8f794fbd32d41d8119a30fc25ca5aaf063f59566 Mon Sep 17 00:00:00 2001 From: Eric House Date: Wed, 1 Jul 2015 07:10:45 -0700 Subject: [PATCH 004/224] beginning of support for invite-by-relay (which is mostly interesting for Rematch): works for linux version, provided you know the relayID of the device you're inviting. Added to common/ a stream-saving version of java's NetLaunchInfo I'll probably want to use there too for cross-platform compatibility (there being no jni support for json.) --- xwords4/common/comms.c | 25 ++- xwords4/common/comms.h | 4 +- xwords4/common/config.mk | 2 + xwords4/common/game.h | 1 + xwords4/common/invit.c | 157 ++++++++++++++++ xwords4/common/invit.h | 75 ++++++++ xwords4/common/nwgamest.c | 1 + xwords4/linux/Makefile | 1 + xwords4/linux/gtkboard.c | 165 +++++++++++------ xwords4/linux/gtkinvit.c | 378 ++++++++++++++++++++++++++++++++++++++ xwords4/linux/gtkinvit.h | 31 ++++ xwords4/linux/gtkmain.c | 66 ++++++- xwords4/linux/gtkutils.c | 2 +- xwords4/linux/gtkutils.h | 2 +- xwords4/linux/linuxmain.c | 15 +- xwords4/linux/linuxmain.h | 1 + xwords4/linux/relaycon.c | 64 +++++++ xwords4/linux/relaycon.h | 6 +- 18 files changed, 919 insertions(+), 77 deletions(-) create mode 100644 xwords4/common/invit.c create mode 100644 xwords4/common/invit.h create mode 100644 xwords4/linux/gtkinvit.c create mode 100644 xwords4/linux/gtkinvit.h diff --git a/xwords4/common/comms.c b/xwords4/common/comms.c index 3151305ab..516db51ed 100644 --- a/xwords4/common/comms.c +++ b/xwords4/common/comms.c @@ -2697,9 +2697,14 @@ countAddrRecs( const CommsCtxt* comms ) XP_Bool addr_iter( const CommsAddrRec* addr, CommsConnType* typp, XP_U32* state ) +{ + return types_iter( addr->_conTypes, typp, state ); +} + +XP_Bool +types_iter( XP_U32 conTypes, CommsConnType* typp, XP_U32* state ) { CommsConnType typ = *state; - XP_U16 conTypes = addr->_conTypes; while ( ++typ < COMMS_CONN_NTYPES ) { *state = typ; XP_U16 mask = 1 << (typ - 1); @@ -2717,11 +2722,17 @@ addr_iter( const CommsAddrRec* addr, CommsConnType* typp, XP_U32* state ) XP_Bool addr_hasType( const CommsAddrRec* addr, CommsConnType typ ) +{ + return types_hasType( addr->_conTypes, typ ); +} + +XP_Bool +types_hasType( XP_U16 conTypes, CommsConnType typ ) { /* Any address has NONE */ XP_Bool hasType = COMMS_CONN_NONE == typ; if ( !hasType ) { - hasType = 0 != (addr->_conTypes & (1 << (typ - 1))); + hasType = 0 != (conTypes & (1 << (typ - 1))); } // XP_LOGF( "%s(%s) => %d", __func__, ConnType2Str(typ), hasType ); return hasType; @@ -2741,11 +2752,17 @@ addr_getType( const CommsAddrRec* addr ) } void -addr_addType( CommsAddrRec* addr, CommsConnType type ) +types_addType( XP_U16* conTypes, CommsConnType type ) { XP_ASSERT( COMMS_CONN_NONE != type ); // XP_LOGF( "%s(%s)", __func__, ConnType2Str(type) ); - addr->_conTypes |= 1 << (type - 1); + *conTypes |= 1 << (type - 1); +} + +void +addr_addType( CommsAddrRec* addr, CommsConnType type ) +{ + types_addType( &addr->_conTypes, type ); } void diff --git a/xwords4/common/comms.h b/xwords4/common/comms.h index b6615294f..e49ddd46b 100644 --- a/xwords4/common/comms.h +++ b/xwords4/common/comms.h @@ -240,11 +240,13 @@ XP_Bool comms_isConnected( const CommsCtxt* const comms ); CommsConnType addr_getType( const CommsAddrRec* addr ); void addr_setType( CommsAddrRec* addr, CommsConnType type ); void addr_addType( CommsAddrRec* addr, CommsConnType type ); +void types_addType( XP_U16* conTypes, CommsConnType type ); void addr_rmType( CommsAddrRec* addr, CommsConnType type ); XP_Bool addr_hasType( const CommsAddrRec* addr, CommsConnType type ); +XP_Bool types_hasType( XP_U16 conTypes, CommsConnType type ); XP_Bool addr_iter( const CommsAddrRec* addr, CommsConnType* typp, XP_U32* state ); - +XP_Bool types_iter( XP_U32 conTypes, CommsConnType* typp, XP_U32* state ); # ifdef DEBUG void comms_getStats( CommsCtxt* comms, XWStreamCtxt* stream ); diff --git a/xwords4/common/config.mk b/xwords4/common/config.mk index 3aad252d1..be10c90ab 100644 --- a/xwords4/common/config.mk +++ b/xwords4/common/config.mk @@ -40,6 +40,7 @@ COMMONSRC = \ $(COMMONDIR)/engine.c \ $(COMMONDIR)/memstream.c \ $(COMMONDIR)/comms.c \ + $(COMMONDIR)/invit.c \ $(COMMONDIR)/mempool.c \ $(COMMONDIR)/movestak.c \ $(COMMONDIR)/strutils.c \ @@ -74,6 +75,7 @@ COMMON4 = \ $(COMMONOBJDIR)/dragdrpp.o \ $(COMMONOBJDIR)/memstream.o \ $(COMMONOBJDIR)/comms.o \ + $(COMMONOBJDIR)/invit.o \ $(COMMONOBJDIR)/mempool.o \ COMMON5 = \ diff --git a/xwords4/common/game.h b/xwords4/common/game.h index 210826f8a..857f70daa 100644 --- a/xwords4/common/game.h +++ b/xwords4/common/game.h @@ -21,6 +21,7 @@ #ifndef _GAME_H_ #define _GAME_H_ +#include "comtypes.h" #include "gameinfo.h" #include "model.h" #include "board.h" diff --git a/xwords4/common/invit.c b/xwords4/common/invit.c new file mode 100644 index 000000000..0d0c34b6f --- /dev/null +++ b/xwords4/common/invit.c @@ -0,0 +1,157 @@ +/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */ +/* + * Copyright 2001 - 2015 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 "invit.h" +#include "comms.h" +#include "strutils.h" + +void +invit_init( InviteInfo* invit, const CurGameInfo* gi, const CommsAddrRec* addr, + XP_U16 nPlayers, XP_U16 forceChannel ) +{ + XP_MEMSET( invit, 0, sizeof(*invit) ); + invit->gameID = gi->gameID; + XP_STRCAT( invit->dict, gi->dictName ); + invit->lang = gi->dictLang; + invit->nPlayersT = gi->nPlayers; + invit->nPlayersH = nPlayers; + invit->forceChannel = forceChannel; + + if ( addr_hasType( addr, COMMS_CONN_RELAY ) ) { + types_addType( &invit->_conTypes, COMMS_CONN_RELAY ); + XP_STRCAT( invit->room, addr->u.ip_relay.invite ); + } + +} + +static XP_U32 +gameID( const InviteInfo* invit ) +{ + XP_U32 gameID = invit->gameID; + if ( 0 == gameID ) { + sscanf( invit->inviteID, "%X", &gameID ); + } + return gameID; +} + +void +invit_setDevID( InviteInfo* invit, XP_U32 devID ) +{ + invit->devID = devID; +} + +void +invit_saveToStream( const InviteInfo* invit, XWStreamCtxt* stream ) +{ + LOG_FUNC(); + stream_putU8( stream, invit->version ); + stream_putU16( stream, invit->_conTypes ); + stream_putU16( stream, invit->lang ); + stringToStream( stream, invit->dict ); + stringToStream( stream, invit->gameName ); + stream_putU8( stream, invit->nPlayersT ); + stream_putU8( stream, invit->nPlayersH ); + stream_putU32( stream, gameID( invit ) ); + stream_putU8( stream, invit->forceChannel ); + + if ( types_hasType( invit->_conTypes, COMMS_CONN_RELAY ) ) { + stringToStream( stream, invit->room ); + XP_LOGF( "%s: wrote room %s", __func__, invit->room ); + stringToStream( stream, invit->inviteID ); + stream_putU32( stream, invit->devID ); + XP_LOGF( "%s: wrote devID %d", __func__, invit->devID ); + } + if ( types_hasType( invit->_conTypes, COMMS_CONN_BT ) ) { + stringToStream( stream, invit->btName ); + stringToStream( stream, invit->btAddress ); + } + if ( types_hasType( invit->_conTypes, COMMS_CONN_SMS ) ) { + stringToStream( stream, invit->phone ); + stream_putU8( stream, invit->isGSM ); + stream_putU8( stream, invit->osType ); + stream_putU32( stream, invit->osVers ); + } +} + +XP_Bool +invit_makeFromStream( InviteInfo* invit, XWStreamCtxt* stream ) +{ + LOG_FUNC(); + XP_MEMSET( invit, 0, sizeof(*invit) ); + invit->version = stream_getU8( stream ); + XP_Bool success = 0 == invit->version; + if ( success ) { + invit->_conTypes = stream_getU16( stream ); + invit->lang = stream_getU16( stream ); + stringFromStreamHere( stream, invit->dict, sizeof(invit->dict) ); + stringFromStreamHere( stream, invit->gameName, sizeof(invit->gameName) ); + invit->nPlayersT = stream_getU8( stream ); + invit->nPlayersH = stream_getU8( stream ); + invit->gameID = stream_getU32( stream ); + invit->forceChannel = stream_getU8( stream ); + + if ( types_hasType( invit->_conTypes, COMMS_CONN_RELAY ) ) { + stringFromStreamHere( stream, invit->room, sizeof(invit->room) ); + XP_LOGF( "%s: read room %s", __func__, invit->room ); + stringFromStreamHere( stream, invit->inviteID, sizeof(invit->inviteID) ); + invit->devID = stream_getU32( stream ); + XP_LOGF( "%s: read devID %d", __func__, invit->devID ); + } + if ( types_hasType( invit->_conTypes, COMMS_CONN_BT ) ) { + stringFromStreamHere( stream, invit->btName, sizeof(invit->btName) ); + stringFromStreamHere( stream, invit->btAddress, sizeof(invit->btAddress) ); + } + if ( types_hasType( invit->_conTypes, COMMS_CONN_SMS ) ) { + stringFromStreamHere( stream, invit->phone, sizeof(invit->phone) ); + invit->isGSM = stream_getU8( stream ); + invit->osType= stream_getU8( stream ); + invit->osVers = stream_getU32( stream ); + } + } + return success; +} + +void +invit_makeAddrRec( const InviteInfo* invit, CommsAddrRec* addr ) +{ + XP_U32 state = 0; + CommsConnType type; + while ( types_iter( invit->_conTypes, &type, &state ) ) { + addr_addType( addr, type ); + switch( type ) { + case COMMS_CONN_RELAY: + XP_STRCAT( addr->u.ip_relay.invite, invit->room ); + /* String relayName = XWPrefs.getDefaultRelayHost( context ); */ + /* int relayPort = XWPrefs.getDefaultRelayPort( context ); */ + /* result.setRelayParams( relayName, relayPort, room ); */ + break; + case COMMS_CONN_BT: + XP_STRCAT( addr->u.bt.btAddr.chars, invit->btAddress ); + XP_STRCAT( addr->u.bt.hostName, invit->btName ); + break; + case COMMS_CONN_SMS: + XP_STRCAT( addr->u.sms.phone, invit->phone ); + break; + default: + XP_ASSERT(0); + break; + } + } +} diff --git a/xwords4/common/invit.h b/xwords4/common/invit.h new file mode 100644 index 000000000..def003f28 --- /dev/null +++ b/xwords4/common/invit.h @@ -0,0 +1,75 @@ +/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */ +/* + * Copyright 2001 - 2015 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. + */ + +#ifndef _INVIT_H_ +#define _INVIT_H_ + +// #include "comms.h" +#include "xwstream.h" +#include "game.h" + +#define MAX_GAME_NAME_LEN 64 +#define MAX_DICT_NAME_LEN 32 + +typedef enum {OSType_NONE, OSType_LINUX, OSType_ANDROID, } XP_OSType; + +typedef struct _InviteInfo { + XP_U8 version; /* struct version for backward compatibility */ + XP_U16 _conTypes; + + XP_UCHAR gameName[MAX_GAME_NAME_LEN]; + XP_UCHAR dict[MAX_DICT_NAME_LEN]; + XP_LangCode lang; + XP_U8 forceChannel; + XP_U8 nPlayersT; + XP_U8 nPlayersH; + + /* Relay */ + XP_UCHAR room[MAX_INVITE_LEN + 1]; + XP_U32 devID; + + /* BT */ + XP_UCHAR btName[32]; + XP_UCHAR btAddress[32]; + + // SMS + XP_UCHAR phone[32]; + XP_Bool isGSM; + XP_OSType osType; + XP_U32 osVers; + + XP_U32 gameID; + XP_UCHAR inviteID[32]; +} InviteInfo; + +void +invit_init( InviteInfo* invit, const CurGameInfo* gi, const CommsAddrRec* addr, + XP_U16 nPlayers, XP_U16 forceChannel ); + + +XP_Bool invit_makeFromStream( InviteInfo* invit, XWStreamCtxt* stream ); +void invit_saveToStream( const InviteInfo* invit, XWStreamCtxt* stream ); + +void invit_makeAddrRec( const InviteInfo* invit, CommsAddrRec* addr ); + +void invit_setDevID( InviteInfo* invit, XP_U32 devID ); + + +#endif diff --git a/xwords4/common/nwgamest.c b/xwords4/common/nwgamest.c index d5e24d8af..08c54e236 100644 --- a/xwords4/common/nwgamest.c +++ b/xwords4/common/nwgamest.c @@ -19,6 +19,7 @@ */ +#include "comtypes.h" #include "nwgamest.h" #include "strutils.h" #include "LocalizedStrIncludes.h" diff --git a/xwords4/linux/Makefile b/xwords4/linux/Makefile index 31364fa8b..e39b1dfed 100644 --- a/xwords4/linux/Makefile +++ b/xwords4/linux/Makefile @@ -187,6 +187,7 @@ GTK_OBJS = \ $(BUILD_PLAT_DIR)/gtkpasswdask.o \ $(BUILD_PLAT_DIR)/gtknewgame.o \ $(BUILD_PLAT_DIR)/gtkconnsdlg.o \ + $(BUILD_PLAT_DIR)/gtkinvit.o \ $(BUILD_PLAT_DIR)/gtkutils.o \ $(BUILD_PLAT_DIR)/gtkntilesask.o \ $(BUILD_PLAT_DIR)/gtkaskdict.o \ diff --git a/xwords4/linux/gtkboard.c b/xwords4/linux/gtkboard.c index f73f6c0be..54aac899c 100644 --- a/xwords4/linux/gtkboard.c +++ b/xwords4/linux/gtkboard.c @@ -1,6 +1,6 @@ /* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */ /* - * Copyright 2000-2013 by Eric House (xwords@eehouse.org). All rights + * Copyright 2000-2015 by Eric House (xwords@eehouse.org). All rights * reserved. * * This program is free software; you can redistribute it and/or @@ -55,6 +55,7 @@ #include "game.h" #include "movestak.h" #include "gtkask.h" +#include "gtkinvit.h" #include "gtkaskm.h" #include "gtkchat.h" #include "gtknewgame.h" @@ -969,15 +970,16 @@ static XP_Bool new_game_impl( GtkGameGlobals* globals, XP_Bool fireConnDlg ) { XP_Bool success = XP_FALSE; + CommonGlobals* cGlobals = &globals->cGlobals; CommsAddrRec addr; - if ( !!globals->cGlobals.game.comms ) { - comms_getAddr( globals->cGlobals.game.comms, &addr ); + if ( !!cGlobals->game.comms ) { + comms_getAddr( cGlobals->game.comms, &addr ); } else { comms_getInitialAddr( &addr, RELAY_NAME_DEFAULT, RELAY_PORT_DEFAULT ); } - CurGameInfo* gi = globals->cGlobals.gi; + CurGameInfo* gi = cGlobals->gi; success = newGameDialog( globals, gi, &addr, XP_TRUE, fireConnDlg ); if ( success ) { #ifndef XWFEATURE_STANDALONE_ONLY @@ -991,9 +993,9 @@ new_game_impl( GtkGameGlobals* globals, XP_Bool fireConnDlg ) #endif }; - if ( !game_reset( MEMPOOL &globals->cGlobals.game, gi, - globals->cGlobals.util, - &globals->cGlobals.cp, &procs ) ) { + if ( !game_reset( MEMPOOL &cGlobals->game, gi, + cGlobals->util, + &cGlobals->cp, &procs ) ) { /* if ( NULL == globals->draw ) { */ /* globals->draw = (GtkDrawCtx*)gtkDrawCtxtMake( globals->drawing_area, */ /* globals ); */ @@ -1013,24 +1015,24 @@ new_game_impl( GtkGameGlobals* globals, XP_Bool fireConnDlg ) } #ifndef XWFEATURE_STANDALONE_ONLY - if ( !!globals->cGlobals.game.comms ) { - comms_setAddr( globals->cGlobals.game.comms, &addr ); + if ( !!cGlobals->game.comms ) { + comms_setAddr( cGlobals->game.comms, &addr ); } else if ( gi->serverRole != SERVER_STANDALONE ) { XP_ASSERT(0); } if ( isClient ) { XWStreamCtxt* stream = - mem_stream_make( MEMPOOL globals->cGlobals.params->vtMgr, - &globals->cGlobals, CHANNEL_NONE, + mem_stream_make( MEMPOOL cGlobals->params->vtMgr, + cGlobals, CHANNEL_NONE, sendOnClose ); - (void)server_initClientConnection( globals->cGlobals.game.server, + (void)server_initClientConnection( cGlobals->game.server, stream ); } #endif - (void)server_do( globals->cGlobals.game.server ); /* assign tiles, etc. */ - board_invalAll( globals->cGlobals.game.board ); - board_draw( globals->cGlobals.game.board ); + (void)server_do( cGlobals->game.server ); /* assign tiles, etc. */ + board_invalAll( cGlobals->game.board ); + board_draw( cGlobals->game.board ); } return success; } /* new_game_impl */ @@ -1559,63 +1561,108 @@ handle_commit_button( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) static void handle_invite_button( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) { - const CurGameInfo* gi = globals->cGlobals.gi; + CommonGlobals* cGlobals = &globals->cGlobals; + const CurGameInfo* gi = cGlobals->gi; - gchar* countStr; - gchar* phone = NULL; - gchar* portstr = NULL; - gchar* forceChannelStr; - AskMInfo infos[] = { - { "Number of players", &countStr }, - { "Remote phone#", &phone }, - { "Remote port", &portstr }, - { "Force channel", &forceChannelStr }, - }; + /* gchar* countStr; */ + /* gchar* phone = NULL; */ + /* gchar* portstr = NULL; */ + /* gchar* forceChannelStr; */ + /* AskMInfo infos[] = { */ + /* { "Number of players", &countStr }, */ + /* { "Remote phone#", &phone }, */ + /* { "Remote port", &portstr }, */ + /* { "Force channel", &forceChannelStr }, */ + /* }; */ XP_U16 nMissing = server_getPendingRegs( globals->cGlobals.game.server ); - gchar buf[64]; - sprintf( buf, "%d", nMissing ); - countStr = buf; - gchar forceChannelBuf[64]; - sprintf( forceChannelBuf, "%d", 1 ); - forceChannelStr = forceChannelBuf; + /* gchar buf[64]; */ + /* sprintf( buf, "%d", nMissing ); */ + /* countStr = buf; */ + /* gchar forceChannelBuf[64]; */ + /* sprintf( forceChannelBuf, "%d", 1 ); */ + /* forceChannelStr = forceChannelBuf; */ - while ( gtkaskm( "Invite how many and how?", infos, VSIZE(infos) ) ) { - int nPlayers = atoi( countStr ); - if ( 0 >= nPlayers || nPlayers > nMissing ) { - gchar buf[128]; - sprintf( buf, "Please invite between 1 and %d players (inclusive).", - nMissing ); - gtktell( globals->window, buf ); - break; - } - - int port = atoi( portstr ); - if ( 0 == port ) { - gtktell( globals->window, "Port must be a number and not 0." ); - break; - } - int forceChannel = atoi( forceChannelStr ); - if ( 1 > forceChannel || 4 <= forceChannel ) { - gtktell( globals->window, "Channel must be between 1 and the number of client devices." ); - break; - } + CommsAddrRec inviteAddr = {0}; + gint nPlayers = nMissing; + XP_U32 devID; + XP_Bool confirmed = gtkInviteDlg( globals, &inviteAddr, &nPlayers, &devID ); + XP_LOGF( "%s: inviteDlg => %d", __func__, confirmed ); + if ( confirmed ) { gchar gameName[64]; snprintf( gameName, VSIZE(gameName), "Game %d", gi->gameID ); CommsAddrRec addr; - CommsCtxt* comms = globals->cGlobals.game.comms; + CommsCtxt* comms = cGlobals->game.comms; XP_ASSERT( comms ); comms_getAddr( comms, &addr ); - linux_sms_invite( globals->cGlobals.params, gi, &addr, gameName, - nPlayers, forceChannel, phone, port ); - break; - } - for ( int ii = 0; ii < VSIZE(infos); ++ii ) { - g_free( *infos[ii].result ); + gint forceChannel = 0; /* PENDING */ + + InviteInfo invit = {0}; + invit_init( &invit, gi, &addr, nPlayers, forceChannel ); + invit_setDevID( &invit, linux_getDevIDRelay( cGlobals->params ) ); + +#ifdef DEBUG + { + XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) + cGlobals->params->vtMgr, + NULL, CHANNEL_NONE, NULL ); + invit_saveToStream( &invit, stream ); + InviteInfo tmp; + invit_makeFromStream( &tmp, stream ); + stream_destroy( stream ); + XP_ASSERT( 0 == memcmp( &invit, &tmp, sizeof(invit) ) ); + } +#endif + + if ( addr_hasType( &inviteAddr, COMMS_CONN_SMS ) ) { + linux_sms_invite( cGlobals->params, gi, &addr, gameName, + nPlayers, forceChannel, + inviteAddr.u.sms.phone, inviteAddr.u.sms.port ); + } + if ( addr_hasType( &addr, COMMS_CONN_RELAY ) ) { + relaycon_invite( cGlobals->params, devID, &invit ); + } } + + /* while ( gtkaskm( "Invite how many and how?", infos, VSIZE(infos) ) ) { */ + /* int nPlayers = atoi( countStr ); */ + /* if ( 0 >= nPlayers || nPlayers > nMissing ) { */ + /* gchar buf[128]; */ + /* sprintf( buf, "Please invite between 1 and %d players (inclusive).", */ + /* nMissing ); */ + /* gtktell( globals->window, buf ); */ + /* break; */ + /* } */ + + /* int port = atoi( portstr ); */ + /* if ( 0 == port ) { */ + /* gtktell( globals->window, "Port must be a number and not 0." ); */ + /* break; */ + /* } */ + /* int forceChannel = atoi( forceChannelStr ); */ + /* if ( 1 > forceChannel || 4 <= forceChannel ) { */ + /* gtktell( globals->window, "Channel must be between 1 and the number of client devices." ); */ + /* break; */ + /* } */ + + /* gchar gameName[64]; */ + /* snprintf( gameName, VSIZE(gameName), "Game %d", gi->gameID ); */ + + /* CommsAddrRec addr; */ + /* CommsCtxt* comms = globals->cGlobals.game.comms; */ + /* XP_ASSERT( comms ); */ + /* comms_getAddr( comms, &addr ); */ + + /* linux_sms_invite( globals->cGlobals.params, gi, &addr, gameName, */ + /* nPlayers, forceChannel, phone, port ); */ + /* break; */ + /* } */ + /* for ( int ii = 0; ii < VSIZE(infos); ++ii ) { */ + /* g_free( *infos[ii].result ); */ + /* } */ } /* handle_invite_button */ static void diff --git a/xwords4/linux/gtkinvit.c b/xwords4/linux/gtkinvit.c new file mode 100644 index 000000000..2aa9d5cea --- /dev/null +++ b/xwords4/linux/gtkinvit.c @@ -0,0 +1,378 @@ +/* -*- compile-command: "make MEMDEBUG=TRUE -j5"; -*- */ +/* + * Copyright 2001-2014 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_GTK + +#include "gtkinvit.h" +#include "gtkutils.h" +#include "linuxbt.h" +#include "comtypes.h" + +typedef struct _PageData { + CommsConnType pageType; + gboolean doUse; + const char* labelText; + GtkWidget* label; + const char* okButtonTxt; +} PageData; + +typedef struct _GtkInviteState { + GtkGameGlobals* globals; + CommsAddrRec* addr; + gint* nPlayersP; + XP_U32* devIDP; + gint maxPlayers; + XP_UCHAR devIDBuf[32]; + + GtkWidget* nPlayersCombo; + /* relay */ + GtkWidget* devID; + /* BT */ + GtkWidget* bthost; + /* SMS */ + GtkWidget* smsphone; + GtkWidget* smsport; + + GtkWidget* bgScanButton; + GtkWidget* okButton; + + GtkWidget* notebook; + guint curPage; + + XP_U16 nTypes; + PageData pageData[COMMS_CONN_NTYPES]; + + gboolean cancelled; +} GtkInviteState; + +/* Make it static so we remember user's late entry */ + +static gint +conTypeToPageNum( const GtkInviteState* state, CommsConnType conType ) +{ + gint pageNum = 0; /* default */ + int ii; + for ( ii = 0; ; ++ii ) { + const PageData* pageData = &state->pageData[ii]; + CommsConnType thisType = pageData->pageType; + if ( thisType == COMMS_CONN_NONE || thisType == conType ) { + pageNum = ii; + break; + } + XP_ASSERT( ii < VSIZE(state->pageData) ); + } + return pageNum; +} + +static void +handle_ok( GtkWidget* XP_UNUSED(widget), gpointer closure ) +{ + GtkInviteState* state = (GtkInviteState*)closure; + const gchar* txt; + guint curPage = state->curPage; + PageData* data = &state->pageData[curPage]; + CommsConnType conType = data->pageType; + + addr_addType( state->addr, conType ); + switch ( conType ) { +#ifdef XWFEATURE_RELAY + case COMMS_CONN_RELAY: + txt = gtk_entry_get_text( GTK_ENTRY(state->devID) ); + *(state->devIDP) = atoi( txt ); + break; +#endif +#ifdef XWFEATURE_BLUETOOTH + case COMMS_CONN_BT: + txt = gtk_entry_get_text( GTK_ENTRY(state->bthost) ); + XP_STRNCPY( state->addr->u.bt.hostName, txt, + sizeof(state->addr->u.bt.hostName) ); + break; +#endif + case COMMS_CONN_SMS: + txt = gtk_entry_get_text( GTK_ENTRY(state->smsphone) ); + XP_STRNCPY( state->addr->u.sms.phone, txt, + sizeof(state->addr->u.sms.phone) ); + txt = gtk_entry_get_text( GTK_ENTRY(state->smsport) ); + state->addr->u.sms.port = atoi( txt ); + break; + default: + XP_ASSERT( 0 ); /* keep compiler happy */ + break; + } + + /* get the number to invite */ + gchar* num = + gtk_combo_box_get_active_text( GTK_COMBO_BOX(state->nPlayersCombo) ); + *(state->nPlayersP) = atoi( num ); + XP_LOGF( "num players: %d", *(state->nPlayersP) ); + + state->cancelled = XP_FALSE; + gtk_main_quit(); +} /* handle_ok */ + +static void +handle_scan( GtkWidget* XP_UNUSED(widget), gpointer closure ) +{ + GtkInviteState* state = (GtkInviteState*)closure; + XP_USE(state); + LOG_FUNC(); + + GSList* devNames = linux_bt_scan(); + if ( !devNames ) { + XP_LOGF( "%s: got nothing", __func__ ); + } else { + GSList* iter; + for ( iter = devNames; !!iter; iter = iter->next ) { +#ifdef DEBUG + gchar* name = iter->data; + XP_LOGF( "%s: got %s", __func__, name ); +#endif + } + } +} + +static void +handle_cancel( GtkWidget* XP_UNUSED(widget), void* closure ) +{ + GtkInviteState* state = (GtkInviteState*)closure; + state->cancelled = XP_TRUE; + gtk_main_quit(); +} + +/* + * Invite: _____ + * Relay: _____ + * Port: _____ + * Cancel OK + */ + +static GtkWidget* +makeRelayPage( GtkInviteState* state, PageData* data ) +{ + data->okButtonTxt = "Invite via Relay"; + + GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); + GtkWidget* hbox; + + hbox = makeLabeledField( "Invitee DeviceID", &state->devID, NULL ); + XP_Bool hasRelay = addr_hasType( state->addr, COMMS_CONN_RELAY ); + if ( hasRelay ) { + gtk_entry_set_text( GTK_ENTRY(state->devID), state->devIDBuf ); + } + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + gtk_widget_show( vbox ); + + return vbox; +} /* makeRelayPage */ + +static GtkWidget* +makeBTPage( GtkInviteState* state, PageData* data ) +{ + data->okButtonTxt = "Invite via Bluetooth"; + + GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); + + GtkWidget* hbox = makeLabeledField( "Invitee device", &state->bthost, NULL ); + if ( addr_hasType( state->addr, data->pageType ) ) { + gtk_entry_set_text( GTK_ENTRY(state->bthost), state->addr->u.bt.hostName ); + } + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + state->bgScanButton = makeButton( "Scan", GTK_SIGNAL_FUNC(handle_scan), + state ); + gtk_box_pack_start( GTK_BOX(vbox), state->bgScanButton, FALSE, TRUE, 0 ); + + gtk_widget_show( vbox ); + + return vbox; +} /* makeBTPage */ + +/* #ifdef XWFEATURE_DIRECTIP */ +/* static GtkWidget* */ +/* makeIPDirPage( GtkInviteState* state, PageData* data ) */ +/* { */ +/* GtkWidget* vbox = boxWithUseCheck( state, data ); */ + +/* /\* XP_UCHAR hostName_ip[MAX_HOSTNAME_LEN + 1]; *\/ */ +/* /\* XP_U16 port_ip; *\/ */ + +/* XP_Bool hasIP = addr_hasType( state->addr, data->pageType ); */ +/* const gchar* name = hasIP ? */ +/* state->addr->u.ip.hostName_ip : state->globals->cGlobals.params->connInfo.ip.hostName; */ +/* GtkWidget* hbox = makeLabeledField( "Hostname", &state->iphost, name ); */ +/* gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); */ + +/* hbox = makeLabeledField( "Relay port", &state->ipport, NULL ); */ +/* if ( hasIP ) { */ +/* char buf[16]; */ +/* snprintf( buf, sizeof(buf), "%d", state->addr->u.ip.port_ip ); */ +/* gtk_entry_set_text( GTK_ENTRY(state->ipport), buf ); */ +/* } */ +/* gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); */ + +/* return vbox; */ +/* } */ +/* #endif */ + +static GtkWidget* +makeSMSPage( GtkInviteState* state, PageData* data ) +{ + data->okButtonTxt = "Invite via SMS"; + + GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); + XP_Bool hasSMS = addr_hasType( state->addr, data->pageType ); + const gchar* phone = hasSMS ? + state->addr->u.sms.phone : state->globals->cGlobals.params->connInfo.sms.phone; + GtkWidget* hbox = makeLabeledField( "Invitee phone", &state->smsphone, phone ); + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + int portVal = hasSMS ? state->addr->u.sms.port + : state->globals->cGlobals.params->connInfo.sms.port; + gchar port[32]; + snprintf( port, sizeof(port), "%d", portVal ); + hbox = makeLabeledField( "Invitee port", &state->smsport, port ); + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + gtk_widget_show( vbox ); + + return vbox; +} /* makeBTPage */ + +static PageData* +getNextData( GtkInviteState* state, CommsConnType typ, gchar* label ) +{ + PageData* result = &state->pageData[state->nTypes++]; + result->pageType = typ; + result->label = gtk_label_new( label ); + result->labelText = label; + return result; +} + +static void +onPageChanged( GtkNotebook* XP_UNUSED(notebook), gpointer XP_UNUSED(arg1), + guint arg2, gpointer data ) +{ + GtkInviteState* state = (GtkInviteState*)data; + state->curPage = arg2; + PageData* pageData = &state->pageData[arg2]; + gtk_button_set_label(GTK_BUTTON(state->okButton), pageData->okButtonTxt ); +} + +XP_Bool +gtkInviteDlg( GtkGameGlobals* globals, CommsAddrRec* addr, gint* nPlayers, + XP_U32* devIDP ) +{ + GtkInviteState state; + XP_MEMSET( &state, 0, sizeof(state) ); + + state.globals = globals; + state.addr = addr; + state.nPlayersP = nPlayers; + state.maxPlayers = *nPlayers; + state.devIDP = devIDP; + + GtkWidget* dialog; + GtkWidget* hbox; + GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); + + hbox = gtk_hbox_new( FALSE, 0 ); + GtkWidget* label = gtk_label_new( "Invite how many:" ); + gtk_box_pack_start( GTK_BOX(hbox), label, FALSE, TRUE, 0 ); + + state.nPlayersCombo = gtk_combo_box_new_text(); + for ( int ii = 1; ii <= state.maxPlayers; ++ii ) { + gchar buf[8]; + sprintf( buf, "%d", ii ); + gtk_combo_box_append_text( GTK_COMBO_BOX(state.nPlayersCombo), buf ); + } + gtk_combo_box_set_active( GTK_COMBO_BOX(state.nPlayersCombo), 0 ); + gtk_box_pack_start( GTK_BOX(hbox), state.nPlayersCombo, FALSE, TRUE, 0 ); + gtk_widget_show( state.nPlayersCombo ); + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + state.notebook = gtk_notebook_new(); + g_signal_connect( GTK_OBJECT(state.notebook), "switch-page", + G_CALLBACK(onPageChanged), &state ); + + PageData* data; + +#ifdef XWFEATURE_RELAY + data = getNextData( &state, COMMS_CONN_RELAY, "Relay" ); + (void)gtk_notebook_append_page( GTK_NOTEBOOK(state.notebook), + makeRelayPage( &state, data ), + data->label ); +#endif +#ifdef XWFEATURE_BLUETOOTH + data = getNextData( &state, COMMS_CONN_BT, "Bluetooth" ); + (void)gtk_notebook_append_page( GTK_NOTEBOOK(state.notebook), + makeBTPage( &state, data ), + data->label ); +#endif +/* #ifdef XWFEATURE_DIRECTIP */ +/* data = getNextData( &state, COMMS_CONN_IP_DIRECT, "Direct" ); */ +/* (void)gtk_notebook_append_page( GTK_NOTEBOOK(state.notebook), */ +/* makeIPDirPage(&state, data), */ +/* data->label ); */ +/* #endif */ +#ifdef XWFEATURE_SMS + data = getNextData( &state, COMMS_CONN_SMS, "SMS" ); + (void)gtk_notebook_append_page( GTK_NOTEBOOK(state.notebook), + makeSMSPage( &state, data ), + data->label ); +#endif + + gtk_box_pack_start( GTK_BOX(vbox), state.notebook, FALSE, TRUE, 0 ); + + /* Set page to the first we actually have */ + XP_U32 st = 0; + CommsConnType firstType; + if ( addr_iter( addr, &firstType, &st ) ) { + gint pageNo = conTypeToPageNum( &state, firstType ); + gtk_notebook_set_current_page( GTK_NOTEBOOK(state.notebook), pageNo ); + } + + gtk_widget_show( state.notebook ); + + /* buttons at the bottom */ + hbox = gtk_hbox_new( FALSE, 0 ); + state.okButton = makeButton( state.pageData[0].okButtonTxt, + GTK_SIGNAL_FUNC(handle_ok), &state ); + gtk_box_pack_start( GTK_BOX(hbox), state.okButton, FALSE, TRUE, 0 ); + gtk_box_pack_start( GTK_BOX(hbox), + makeButton( "Cancel", + GTK_SIGNAL_FUNC(handle_cancel), + &state ), + FALSE, TRUE, 0 ); + gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); + + gtk_widget_show( vbox ); + + dialog = gtk_dialog_new(); + gtk_window_set_modal( GTK_WINDOW( dialog ), TRUE ); + gtk_container_add( GTK_CONTAINER( GTK_DIALOG(dialog)->action_area), vbox ); + + + gtk_widget_show_all( dialog ); + gtk_main(); + gtk_widget_destroy( dialog ); + + return !state.cancelled; +} /* gtkInviteDlg */ +#endif diff --git a/xwords4/linux/gtkinvit.h b/xwords4/linux/gtkinvit.h new file mode 100644 index 000000000..ba2b34596 --- /dev/null +++ b/xwords4/linux/gtkinvit.h @@ -0,0 +1,31 @@ +/* -*- compile-command: "make MEMDEBUG=TRUE -j5"; -*- */ +/* + * Copyright 2001-2014 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_GTK + +#include "xptypes.h" +#include "gtkboard.h" + +/* return true if not cancelled */ +XP_Bool gtkInviteDlg( GtkGameGlobals* globals, CommsAddrRec* addr, + /*inout*/ gint* nPlayers, XP_U32* devID ); + +#endif + + diff --git a/xwords4/linux/gtkmain.c b/xwords4/linux/gtkmain.c index ed85a8670..c0fc65c9f 100644 --- a/xwords4/linux/gtkmain.c +++ b/xwords4/linux/gtkmain.c @@ -336,16 +336,12 @@ addButton( gchar* label, GtkWidget* parent, GCallback proc, void* closure ) return button; } -static GtkWidget* -makeGamesWindow( GtkAppGlobals* apg ) +static void +setWindowTitle( GtkAppGlobals* apg ) { - GtkWidget* window; + GtkWidget* window = apg->window; LaunchParams* params = apg->params; - window = gtk_window_new( GTK_WINDOW_TOPLEVEL ); - g_signal_connect( G_OBJECT(window), "destroy", - G_CALLBACK(handle_destroy), apg ); - gchar title[128] = {0}; if ( !!params->dbName ) { strcat( title, params->dbName ); @@ -354,8 +350,26 @@ makeGamesWindow( GtkAppGlobals* apg ) int len = strlen( title ); snprintf( &title[len], VSIZE(title) - len, " (phone: %s, port: %d)", params->connInfo.sms.phone, params->connInfo.sms.port ); +#endif +#ifdef XWFEATURE_RELAY + XP_U32 relayID = linux_getDevIDRelay( params ); + len = strlen( title ); + snprintf( &title[len], VSIZE(title) - len, " (relayid: %d)", relayID ); #endif gtk_window_set_title( GTK_WINDOW(window), title ); +} + +static void +makeGamesWindow( GtkAppGlobals* apg ) +{ + GtkWidget* window; + LaunchParams* params = apg->params; + + apg->window = window = gtk_window_new( GTK_WINDOW_TOPLEVEL ); + g_signal_connect( G_OBJECT(window), "destroy", + G_CALLBACK(handle_destroy), apg ); + + setWindowTitle( apg ); GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); gtk_container_add( GTK_CONTAINER(window), vbox ); @@ -388,7 +402,6 @@ makeGamesWindow( GtkAppGlobals* apg ) updateButtons( apg ); gtk_widget_show( window ); - return window; } static GtkWidget* @@ -459,6 +472,38 @@ gtkSocketAdded( void* closure, int newSock, GIOFunc proc ) LOG_RETURN_VOID(); } /* gtk_socket_changed */ + +static void +relayInviteReceived( void* closure, InviteInfo* invite ) +{ + GtkAppGlobals* apg = (GtkAppGlobals*)closure; + LaunchParams* params = apg->params; + + CurGameInfo gi = {0}; + gi_copy( MPPARM(params->mpool) &gi, ¶ms->pgi ); + + gi_setNPlayers( &gi, invite->nPlayersT, invite->nPlayersH ); + gi.gameID = invite->gameID; + gi.dictLang = invite->lang; + gi.forceChannel = invite->forceChannel; + replaceStringIfDifferent( params->mpool, &gi.dictName, invite->dict ); + + GtkGameGlobals* globals = malloc( sizeof(*globals) ); + params->needsNewGame = XP_FALSE; + initGlobals( globals, params, &gi ); + + invit_makeAddrRec( invite, &globals->cGlobals.addr ); + // globals->cGlobals.addr = *returnAddr; + + GtkWidget* gameWindow = globals->window; + globals->cGlobals.pDb = apg->params->pDb; + globals->cGlobals.selRow = -1; + recordOpened( apg, globals ); + gtk_widget_show( gameWindow ); + + gi_disposePlayerInfo( MPPARM(params->mpool) &gi ); +} + static void gtkGotBuf( void* closure, const CommsAddrRec* from, const XP_U8* buf, XP_U16 len ) @@ -570,6 +615,8 @@ gtkDevIDReceived( void* closure, const XP_UCHAR* devID, XP_U16 maxInterval ) XP_LOGF( "%s(devID=%s)", __func__, devID ); db_store( params->pDb, KEY_RDEVID, devID ); (void)g_timeout_add_seconds( maxInterval, keepalive_timer, apg ); + + setWindowTitle( apg ); } else { XP_LOGF( "%s: bad relayid", __func__ ); db_remove( params->pDb, KEY_RDEVID ); @@ -647,6 +694,7 @@ gtkmain( LaunchParams* params ) .devIDReceived = gtkDevIDReceived, .msgErrorMsg = gtkErrorMsgRcvd, .socketAdded = gtkSocketAdded, + .inviteReceived = relayInviteReceived, }; relaycon_init( params, &procs, &apg, @@ -685,7 +733,7 @@ gtkmain( LaunchParams* params ) #endif - apg.window = makeGamesWindow( &apg ); + makeGamesWindow( &apg ); } else if ( !!params->dbFileName ) { apg.window = openDBFile( &apg ); } diff --git a/xwords4/linux/gtkutils.c b/xwords4/linux/gtkutils.c index 0ed24d55e..a05d6625e 100644 --- a/xwords4/linux/gtkutils.c +++ b/xwords4/linux/gtkutils.c @@ -24,7 +24,7 @@ #include "gtkutils.h" GtkWidget* -makeButton( char* text, GCallback func, gpointer data ) +makeButton( const char* text, GCallback func, gpointer data ) { GtkWidget* button = gtk_button_new_with_label( text ); g_signal_connect( GTK_OBJECT(button), "clicked", func, data ); diff --git a/xwords4/linux/gtkutils.h b/xwords4/linux/gtkutils.h index 07950bb1d..7618aa5f4 100644 --- a/xwords4/linux/gtkutils.h +++ b/xwords4/linux/gtkutils.h @@ -26,7 +26,7 @@ #include "gtkboard.h" -GtkWidget* makeButton( char* text, GCallback func, gpointer data ); +GtkWidget* makeButton( const char* text, GCallback func, gpointer data ); GtkWidget* makeLabeledField( const char* labelText, GtkWidget** field, const gchar* initialVal ); diff --git a/xwords4/linux/linuxmain.c b/xwords4/linux/linuxmain.c index ad74d6575..92d6b57e8 100644 --- a/xwords4/linux/linuxmain.c +++ b/xwords4/linux/linuxmain.c @@ -895,6 +895,19 @@ linShiftFocus( CommonGlobals* cGlobals, XP_Key key, const BoardObjectType* order } /* linShiftFocus */ #endif +const XP_U32 +linux_getDevIDRelay( LaunchParams* params ) +{ + XP_U32 result = 0; + gchar buf[32]; + if ( db_fetch( params->pDb, KEY_RDEVID, buf, sizeof(buf) ) ) { + sscanf( buf, "%X", &result ); + XP_LOGF( "%s(): %s => %x", __func__, buf, result ); + } + LOG_RETURNF( "%d", result ); + return result; +} + const XP_UCHAR* linux_getDevID( LaunchParams* params, DevIDType* typ ) { @@ -2088,7 +2101,7 @@ main( int argc, char** argv ) mainParams.showRobotScores = XP_FALSE; mainParams.useMmap = XP_TRUE; mainParams.useUdp = true; - mainParams.dbName = "xwgames.sql"; + mainParams.dbName = "xwgames.sqldb"; char* envDictPath = getenv( "XW_DICTDIR" ); XP_LOGF( "%s: envDictPath=%s", __func__, envDictPath ); diff --git a/xwords4/linux/linuxmain.h b/xwords4/linux/linuxmain.h index e161093ce..29c5d5f7c 100644 --- a/xwords4/linux/linuxmain.h +++ b/xwords4/linux/linuxmain.h @@ -106,6 +106,7 @@ void gameGotBuf( CommonGlobals* globals, XP_Bool haveDraw, const XP_U8* buf, XP_U16 len, const CommsAddrRec* from ); gboolean app_socket_proc( GIOChannel* source, GIOCondition condition, gpointer data ); +const XP_U32 linux_getDevIDRelay( LaunchParams* params ); const XP_UCHAR* linux_getDevID( LaunchParams* params, DevIDType* typ ); void linux_doInitialReg( LaunchParams* params, XP_Bool idIsNew ); XP_Bool linux_setupDevidParams( LaunchParams* params ); diff --git a/xwords4/linux/relaycon.c b/xwords4/linux/relaycon.c index 2e40c56c0..34a189020 100644 --- a/xwords4/linux/relaycon.c +++ b/xwords4/linux/relaycon.c @@ -22,6 +22,7 @@ #include #include "relaycon.h" +#include "linuxmain.h" #include "comtypes.h" typedef struct _RelayConStorage { @@ -31,6 +32,7 @@ typedef struct _RelayConStorage { struct sockaddr_in saddr; uint32_t nextID; XWPDevProto proto; + LaunchParams* params; } RelayConStorage; typedef struct _MsgHeader { @@ -53,6 +55,7 @@ static int writeHeader( RelayConStorage* storage, XP_U8* dest, XWRelayReg cmd ); static bool readHeader( const XP_U8** buf, MsgHeader* header ); static size_t writeDevID( XP_U8* buf, size_t len, const XP_UCHAR* str ); static size_t writeShort( XP_U8* buf, size_t len, XP_U16 shrt ); +static size_t writeLong( XP_U8* buf, size_t len, XP_U32 lng ); static size_t writeVLI( XP_U8* out, uint32_t nn ); static size_t un2vli( int nn, uint8_t* buf ); static bool vli2un( const uint8_t** inp, uint32_t* outp ); @@ -75,6 +78,8 @@ relaycon_init( LaunchParams* params, const RelayConnProcs* procs, storage->saddr.sin_addr.s_addr = htonl( hostNameToIP(host) ); storage->saddr.sin_port = htons(port); + storage->params = params; + storage->proto = XWPDEV_PROTO_VERSION_1; } @@ -107,6 +112,34 @@ relaycon_reg( LaunchParams* params, const XP_UCHAR* rDevID, sendIt( storage, tmpbuf, indx ); } +void +relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invit ) +{ + XP_U8 tmpbuf[256]; + int indx = 0; + + RelayConStorage* storage = getStorage( params ); + indx += writeHeader( storage, tmpbuf, XWPDEV_INVITE ); + XP_U32 me = linux_getDevIDRelay( params ); + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, me ); + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, dest ); + + XWStreamCtxt* stream = mem_stream_make( MPPARM(params->mpool) + params->vtMgr, params, + CHANNEL_NONE, NULL ); + invit_saveToStream( invit, stream ); + XP_U16 len = stream_getSize( stream ); + indx += writeShort( &tmpbuf[indx], sizeof(tmpbuf) - indx, len ); + XP_ASSERT( indx + len < sizeof(tmpbuf) ); + const XP_U8* ptr = stream_getPtr( stream ); + XP_MEMCPY( &tmpbuf[indx], ptr, len ); + indx += len; + stream_destroy( stream ); + + sendIt( storage, tmpbuf, indx ); + LOG_RETURN_VOID(); +} + XP_S16 relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, XP_U32 gameToken, const CommsAddrRec* XP_UNUSED(addrRec) ) @@ -295,6 +328,27 @@ relaycon_receive( GIOChannel* source, GIOCondition XP_UNUSED_DBG(condition), gpo XP_LOGF( "%s: got message: %s", __func__, buf ); break; } + case XWPDEV_GOTINVITE: { + XP_LOGF( "%s(): got XWPDEV_GOTINVITE", __func__ ); + XP_U32 sender = getNetLong( &ptr ); + XP_U16 len = getNetShort( &ptr ); + XWStreamCtxt* stream = mem_stream_make( MPPARM(storage->params->mpool) + storage->params->vtMgr, storage, + CHANNEL_NONE, NULL ); + stream_putBytes( stream, ptr, len ); + InviteInfo invit; + XP_Bool success = invit_makeFromStream( &invit, stream ); + XP_LOGF( "sender: %d; invit.devID: %d", sender, invit.devID ); + XP_ASSERT( sender == invit.devID ); + stream_destroy( stream ); + + if ( success ) { + (*storage->procs.inviteReceived)( storage->procsClosure, + &invit ); + } + } + break; + default: XP_LOGF( "%s: Unexpected cmd %d", __func__, header.cmd ); XP_ASSERT( 0 ); @@ -388,6 +442,15 @@ writeShort( XP_U8* buf, size_t len, XP_U16 shrt ) return sizeof(shrt); } +static size_t +writeLong( XP_U8* buf, size_t len, XP_U32 lng ) +{ + lng = htonl( lng ); + assert( sizeof( lng ) <= len ); + memcpy( buf, &lng, sizeof(lng) ); + return sizeof(lng); +} + static size_t writeVLI( XP_U8* out, uint32_t nn ) { @@ -430,6 +493,7 @@ writeHeader( RelayConStorage* storage, XP_U8* dest, XWRelayReg cmd ) { int indx = 0; dest[indx++] = storage->proto; + XP_LOGF( "%s: wrote proto %d", __func__, storage->proto ); uint32_t packetNum = 0; if ( XWPDEV_ACK != cmd ) { packetNum = storage->nextID++; diff --git a/xwords4/linux/relaycon.h b/xwords4/linux/relaycon.h index 30ebb1bc3..4a2967659 100644 --- a/xwords4/linux/relaycon.h +++ b/xwords4/linux/relaycon.h @@ -1,6 +1,7 @@ /* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */ /* - * Copyright 2013 by Eric House (xwords@eehouse.org). All rights reserved. + * Copyright 2013 - 2015 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 @@ -21,6 +22,7 @@ #define _RELAYCON_H_ #include "main.h" +#include "invit.h" typedef struct _Procs { void (*msgReceived)( void* closure, const CommsAddrRec* from, @@ -29,6 +31,7 @@ typedef struct _Procs { void (*devIDReceived)( void* closure, const XP_UCHAR* devID, XP_U16 maxInterval ); void (*msgErrorMsg)( void* closure, const XP_UCHAR* msg ); + void (*inviteReceived)( void* closure, InviteInfo* invit ); SocketAddedFunc socketAdded; } RelayConnProcs; @@ -36,6 +39,7 @@ void relaycon_init( LaunchParams* params, const RelayConnProcs* procs, void* procsClosure, const char* host, int port ); void relaycon_reg( LaunchParams* params, const XP_UCHAR* rDevID, DevIDType typ, const XP_UCHAR* devID ); +void relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invite ); XP_S16 relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, XP_U32 gameToken, const CommsAddrRec* addrRec ); XP_S16 relaycon_sendnoconn( LaunchParams* params, const XP_U8* buf, From b451b936e77b13c208cee2cb3d1edf3be61709a1 Mon Sep 17 00:00:00 2001 From: Eric House Date: Wed, 1 Jul 2015 05:52:02 -0700 Subject: [PATCH 005/224] fix assertion failure, probably due to some freak accident of missaved timestamp. Whatever. --- .../XWords4/src/org/eehouse/android/xw4/ExpiringDelegate.java | 2 ++ 1 file changed, 2 insertions(+) diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/ExpiringDelegate.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/ExpiringDelegate.java index 7a783344e..e5b4045b6 100644 --- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/ExpiringDelegate.java +++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/ExpiringDelegate.java @@ -201,6 +201,8 @@ public class ExpiringDelegate { m_pct = (int)((100 * passed) / INTERVAL_SECS); if ( m_pct > 100 ) { m_pct = 100; + } else if ( m_pct < 0 ) { + m_pct = 0; } else if ( null != m_handler ) { long onePct = INTERVAL_SECS / 100; long lastStart = m_startSecs + (onePct * m_pct); From 3a896df199959666aa5989b85fb3d2fd6089e376 Mon Sep 17 00:00:00 2001 From: Eric House Date: Wed, 1 Jul 2015 18:38:33 -0700 Subject: [PATCH 006/224] show game-over dialog (with its Rematch button) when a over game is opened --- xwords4/linux/gtkboard.c | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/xwords4/linux/gtkboard.c b/xwords4/linux/gtkboard.c index 54aac899c..440e15be8 100644 --- a/xwords4/linux/gtkboard.c +++ b/xwords4/linux/gtkboard.c @@ -80,6 +80,8 @@ static void disenable_buttons( GtkGameGlobals* globals ); static GtkWidget* addButton( GtkWidget* hbox, gchar* label, GCallback func, GtkGameGlobals* globals ); static void handle_invite_button( GtkWidget* widget, GtkGameGlobals* globals ); +static void gtkShowFinalScores( const GtkGameGlobals* globals, + XP_Bool ignoreTimeout ); #define GTK_TRAY_HT_ROWS 3 @@ -863,6 +865,14 @@ destroy_board_window( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) // gtk_main_quit(); } +static void +on_board_window_shown( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) +{ + if ( server_getGameIsOver( globals->cGlobals.game.server ) ) { + gtkShowFinalScores( globals, XP_TRUE ); + } +} + static void cleanup( GtkGameGlobals* globals ) { @@ -1769,7 +1779,7 @@ gtk_util_yOffsetChange( XW_UtilCtxt* uc, XP_U16 maxOffset, } /* gtk_util_yOffsetChange */ static void -gtkShowFinalScores( const GtkGameGlobals* globals ) +gtkShowFinalScores( const GtkGameGlobals* globals, XP_Bool ignoreTimeout ) { XWStreamCtxt* stream; XP_UCHAR* text; @@ -1783,7 +1793,8 @@ gtkShowFinalScores( const GtkGameGlobals* globals ) text = strFromStream( stream ); stream_destroy( stream ); - XP_U16 timeout = cGlobals->manualFinal? 0 : cGlobals->params->askTimeout; + XP_U16 timeout = (ignoreTimeout || cGlobals->manualFinal) + ? 0 : cGlobals->params->askTimeout; const AskPair buttons[] = { { "OK", 1 }, { "Rematch", 2 }, @@ -1836,7 +1847,7 @@ gtk_util_notifyGameOver( XW_UtilCtxt* uc, XP_S16 quitter ) server_handleUndo( cGlobals->game.server, 0 ); board_draw( cGlobals->game.board ); } else if ( !cGlobals->params->skipGameOver ) { - gtkShowFinalScores( globals ); + gtkShowFinalScores( globals, XP_FALSE ); } } /* gtk_util_notifyGameOver */ @@ -2623,6 +2634,9 @@ initGlobals( GtkGameGlobals* globals, LaunchParams* params, CurGameInfo* gi ) g_signal_connect( G_OBJECT (window), "destroy", G_CALLBACK( destroy_board_window ), globals ); + XP_ASSERT( !!globals ); + g_signal_connect( G_OBJECT (window), "show", + G_CALLBACK( on_board_window_shown ), globals ); menubar = makeMenus( globals ); gtk_box_pack_start( GTK_BOX(vbox), menubar, FALSE, TRUE, 0); From 33d0db93e8bd25de83725f2477bed68ad275185a Mon Sep 17 00:00:00 2001 From: Eric House Date: Mon, 6 Jul 2015 20:23:22 -0700 Subject: [PATCH 007/224] add some consts --- xwords4/common/board.c | 2 +- xwords4/common/board.h | 2 +- xwords4/common/server.c | 2 +- xwords4/common/server.h | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/xwords4/common/board.c b/xwords4/common/board.c index 97003fd88..56a56cfbb 100644 --- a/xwords4/common/board.c +++ b/xwords4/common/board.c @@ -609,7 +609,7 @@ board_getScale( BoardCtxt* board, XP_U16* hScale, XP_U16* vScale ) #endif XP_Bool -board_prefsChanged( BoardCtxt* board, CommonPrefs* cp ) +board_prefsChanged( BoardCtxt* board, const CommonPrefs* cp ) { XP_Bool showArrowChanged; XP_Bool hideValChanged; diff --git a/xwords4/common/board.h b/xwords4/common/board.h index 4573fb1be..4b4b41380 100644 --- a/xwords4/common/board.h +++ b/xwords4/common/board.h @@ -156,7 +156,7 @@ XP_Bool board_requestHint( BoardCtxt* board, #endif XP_Bool usePrev, XP_Bool* workRemainsP ); -XP_Bool board_prefsChanged( BoardCtxt* board, CommonPrefs* cp ); +XP_Bool board_prefsChanged( BoardCtxt* board, const CommonPrefs* cp ); BoardObjectType board_getFocusOwner( BoardCtxt* board ); diff --git a/xwords4/common/server.c b/xwords4/common/server.c index 2c8ec9ad5..116e18639 100644 --- a/xwords4/common/server.c +++ b/xwords4/common/server.c @@ -546,7 +546,7 @@ figureSleepTime( const ServerCtxt* server ) #endif void -server_prefsChanged( ServerCtxt* server, CommonPrefs* cp ) +server_prefsChanged( ServerCtxt* server, const CommonPrefs* cp ) { server->nv.showRobotScores = cp->showRobotScores; server->nv.sortNewTiles = cp->sortNewTiles; diff --git a/xwords4/common/server.h b/xwords4/common/server.h index 515709a58..977c44620 100644 --- a/xwords4/common/server.h +++ b/xwords4/common/server.h @@ -67,7 +67,7 @@ void server_writeToStream( const ServerCtxt* server, XWStreamCtxt* stream ); void server_reset( ServerCtxt* server, CommsCtxt* comms ); void server_destroy( ServerCtxt* server ); -void server_prefsChanged( ServerCtxt* server, CommonPrefs* cp ); +void server_prefsChanged( ServerCtxt* server, const CommonPrefs* cp ); typedef void (*TurnChangeListener)( void* data ); void server_setTurnChangeListener( ServerCtxt* server, TurnChangeListener tl, From fd06bca1513e0137cce50bb15e09bf1b5ea474a7 Mon Sep 17 00:00:00 2001 From: Eric House Date: Mon, 6 Jul 2015 20:24:45 -0700 Subject: [PATCH 008/224] assert stream version being set consistently, not changed --- xwords4/common/memstream.c | 2 ++ 1 file changed, 2 insertions(+) diff --git a/xwords4/common/memstream.c b/xwords4/common/memstream.c index 129542f89..fcf62c47b 100644 --- a/xwords4/common/memstream.c +++ b/xwords4/common/memstream.c @@ -396,6 +396,8 @@ static void mem_stream_setVersion( XWStreamCtxt* p_sctx, XP_U16 vers ) { MemStreamCtxt* stream = (MemStreamCtxt*)p_sctx; + /* Something's wrong if we're changing it -- I think */ + XP_ASSERT( 0 == stream->version || vers == stream->version ); stream->version = vers; } /* mem_stream_setVersion */ From 0f332281551226f20f0d95448b2fc1806ac0d9db Mon Sep 17 00:00:00 2001 From: Eric House Date: Mon, 6 Jul 2015 20:28:16 -0700 Subject: [PATCH 009/224] get rematch working on linux, at least for two-device games. (Handling the case where one of several guests wants to rematch is a hard problem for later.) Requires passing old-style relayIDs (connname plus device index) when devIDs aren't available, which they may not always be. --- xwords4/common/comms.c | 45 ++++++---- xwords4/common/comms.h | 5 ++ xwords4/common/comtypes.h | 3 +- xwords4/common/game.c | 24 +++++- xwords4/common/game.h | 5 +- xwords4/common/invit.c | 1 + xwords4/common/invit.h | 6 ++ xwords4/linux/gamesdb.c | 171 +++++++++++++++++++++++++++----------- xwords4/linux/gamesdb.h | 7 ++ xwords4/linux/gtkaskm.c | 4 +- xwords4/linux/gtkboard.c | 122 ++++++++++++++++++--------- xwords4/linux/gtkinvit.c | 11 +-- xwords4/linux/gtkinvit.h | 2 +- xwords4/linux/gtkmain.c | 155 +++++++++++++++++++++++++--------- xwords4/linux/gtkmain.h | 2 + xwords4/linux/linuxmain.c | 4 + xwords4/linux/main.h | 3 +- xwords4/linux/relaycon.c | 62 ++++++++------ xwords4/linux/relaycon.h | 5 +- xwords4/relay/dbmgr.cpp | 13 +++ xwords4/relay/dbmgr.h | 2 + xwords4/relay/xwrelay.cpp | 17 ++-- 22 files changed, 484 insertions(+), 185 deletions(-) diff --git a/xwords4/common/comms.c b/xwords4/common/comms.c index 516db51ed..b0bdbcd37 100644 --- a/xwords4/common/comms.c +++ b/xwords4/common/comms.c @@ -560,6 +560,7 @@ comms_setConnID( CommsCtxt* comms, XP_U32 connID ) static void addrFromStreamOne( CommsAddrRec* addrP, XWStreamCtxt* stream, CommsConnType typ ) { + XP_U16 version = stream_getVersion( stream ); switch( typ ) { case COMMS_CONN_NONE: break; @@ -584,8 +585,11 @@ addrFromStreamOne( CommsAddrRec* addrP, XWStreamCtxt* stream, CommsConnType typ stringFromStreamHere( stream, addrP->u.ip_relay.hostName, sizeof(addrP->u.ip_relay.hostName) ); addrP->u.ip_relay.ipAddr = stream_getU32( stream ); + if ( version >= STREAM_VERS_DEVIDS ) { + addrP->u.ip_relay.devID = stream_getU32( stream ); + } addrP->u.ip_relay.port = stream_getU16( stream ); - if ( stream_getVersion( stream ) >= STREAM_VERS_DICTLANG ) { + if ( version >= STREAM_VERS_DICTLANG ) { addrP->u.ip_relay.seeksPublicRoom = stream_getBits( stream, 1 ); addrP->u.ip_relay.advertiseRoom = stream_getBits( stream, 1 ); } @@ -632,7 +636,8 @@ comms_makeFromStream( MPFORMAL XWStreamCtxt* stream, XW_UtilCtxt* util, isServer = stream_getU8( stream ); addrFromStream( &addr, stream ); - if ( addr_hasType( &addr, COMMS_CONN_RELAY ) ) { + if ( version >= STREAM_VERS_DEVIDS + || addr_hasType( &addr, COMMS_CONN_RELAY ) ) { nPlayersHere = (XP_U16)stream_getBits( stream, 4 ); nPlayersTotal = (XP_U16)stream_getBits( stream, 4 ); } else { @@ -810,6 +815,7 @@ addrToStreamOne( XWStreamCtxt* stream, CommsConnType typ, const CommsAddrRec* ad stringToStream( stream, addrP->u.ip_relay.invite ); stringToStream( stream, addrP->u.ip_relay.hostName ); stream_putU32( stream, addrP->u.ip_relay.ipAddr ); + stream_putU32( stream, addrP->u.ip_relay.devID ); stream_putU16( stream, addrP->u.ip_relay.port ); stream_putBits( stream, 1, addrP->u.ip_relay.seeksPublicRoom ); stream_putBits( stream, 1, addrP->u.ip_relay.advertiseRoom ); @@ -827,6 +833,7 @@ addrToStreamOne( XWStreamCtxt* stream, CommsConnType typ, const CommsAddrRec* ad void addrToStream( XWStreamCtxt* stream, const CommsAddrRec* addrP ) { + stream_setVersion( stream, CUR_STREAM_VERS ); stream_putU8( stream, addrP->_conTypes ); CommsConnType typ; @@ -843,13 +850,13 @@ comms_writeToStream( CommsCtxt* comms, XWStreamCtxt* stream, AddressRecord* rec; MsgQueueElem* msg; + stream_setVersion( stream, CUR_STREAM_VERS ); + stream_putU8( stream, (XP_U8)comms->isServer ); logAddr( comms, &comms->addr, __func__ ); addrToStream( stream, &comms->addr ); - if ( addr_hasType( &comms->addr, COMMS_CONN_RELAY ) ) { - stream_putBits( stream, 4, comms->rr.nPlayersHere ); - stream_putBits( stream, 4, comms->rr.nPlayersTotal ); - } + stream_putBits( stream, 4, comms->rr.nPlayersHere ); + stream_putBits( stream, 4, comms->rr.nPlayersTotal ); stream_putU32( stream, comms->connID ); stream_putU16( stream, comms->nextChannelNo ); @@ -982,6 +989,20 @@ formatRelayID( const CommsCtxt* comms, XWHostID hostID, return XP_TRUE; } +XP_Bool +comms_formatRelayID( const CommsCtxt* comms, XP_U16 indx, + XP_UCHAR* buf, XP_U16* lenp ) +{ + XP_LOGF( "%s(indx=%d)", __func__, indx ); + XWHostID hostID = HOST_ID_SERVER; + if ( comms->isServer ) { + hostID += 1 + indx; + } + XP_Bool success = formatRelayID( comms, hostID, buf, lenp ); + XP_LOGF( "%s(%d) => %s", __func__, indx, buf ); + return success; +} + /* Get *my* "relayID", a combo of connname and host id */ XP_Bool comms_getRelayID( const CommsCtxt* comms, XP_UCHAR* buf, XP_U16* lenp ) @@ -997,12 +1018,14 @@ comms_getInitialAddr( CommsAddrRec* addr #ifdef XWFEATURE_RELAY , const XP_UCHAR* relayName , XP_U16 relayPort + , XP_U32 devID #endif ) { #if defined XWFEATURE_RELAY addr_setType( addr, COMMS_CONN_RELAY ); /* for temporary ease in debugging */ addr->u.ip_relay.ipAddr = 0L; /* force 'em to set it */ + addr->u.ip_relay.devID = devID; addr->u.ip_relay.port = relayPort; { const char* name = relayName; @@ -1051,7 +1074,7 @@ comms_getConTypes( const CommsCtxt* comms ) XP_LOGF( "%s: returning COMMS_CONN_NONE for null comms", __func__ ); } return typ; -} /* comms_getConType */ +} /* comms_getConTypes */ XP_Bool comms_getIsServer( const CommsCtxt* comms ) @@ -2629,14 +2652,8 @@ augmentChannelAddr( CommsCtxt* comms, AddressRecord * const rec, switch( typ ) { case COMMS_CONN_RELAY: dest = &rec->addr.u.ip_relay; + src = &addr->u.ip_relay; siz = sizeof( rec->addr.u.ip_relay ); - /* Special case for relay: use comms' relay info if caller - didn't bother to fill it in */ - if ( addr->u.ip_relay.invite[0] ) { - src = &addr->u.ip_relay; - } else { - src = &comms->addr.u.ip_relay; - } if ( 0 != hostID ) { rec->rr.hostID = hostID; XP_LOGF( "%s: set hostID for rec %p to %d", __func__, rec, hostID ); diff --git a/xwords4/common/comms.h b/xwords4/common/comms.h index e49ddd46b..d300ca261 100644 --- a/xwords4/common/comms.h +++ b/xwords4/common/comms.h @@ -90,6 +90,7 @@ typedef struct _CommsAddrRec { XP_UCHAR invite[MAX_INVITE_LEN + 1]; XP_UCHAR hostName[MAX_HOSTNAME_LEN + 1]; XP_U32 ipAddr; /* looked up from above */ + XP_U32 devID; XP_U16 port; XP_Bool seeksPublicRoom; XP_Bool advertiseRoom; @@ -180,6 +181,7 @@ void comms_getInitialAddr( CommsAddrRec* addr #ifdef XWFEATURE_RELAY , const XP_UCHAR* relayName , XP_U16 relayPort + , XP_U32 devID #endif ); XP_Bool comms_checkAddr( DeviceRole role, const CommsAddrRec* addr, @@ -189,6 +191,9 @@ void comms_getAddr( const CommsCtxt* comms, CommsAddrRec* addr ); void comms_setAddr( CommsCtxt* comms, const CommsAddrRec* addr ); void comms_getAddrs( const CommsCtxt* comms, CommsAddrRec addr[], XP_U16* nRecs ); +XP_Bool comms_formatRelayID( const CommsCtxt* comms, XP_U16 indx, + XP_UCHAR* buf, XP_U16* lenp ); + XP_U16 comms_countPendingPackets( const CommsCtxt* comms ); diff --git a/xwords4/common/comtypes.h b/xwords4/common/comtypes.h index 094ddcbab..cca8f4476 100644 --- a/xwords4/common/comtypes.h +++ b/xwords4/common/comtypes.h @@ -47,6 +47,7 @@ #endif #define MAX_COLS MAX_ROWS +#define STREAM_VERS_DEVIDS 0x19 #define STREAM_VERS_MULTIADDR 0x18 #define STREAM_VERS_MODELDIVIDER 0x17 #define STREAM_VERS_COMMSBACKOFF 0x16 @@ -86,7 +87,7 @@ #define STREAM_VERS_405 0x01 /* search for FIX_NEXT_VERSION_CHANGE next time this is changed */ -#define CUR_STREAM_VERS STREAM_VERS_MULTIADDR +#define CUR_STREAM_VERS STREAM_VERS_DEVIDS typedef struct XP_Rect { XP_S16 left; diff --git a/xwords4/common/game.c b/xwords4/common/game.c index a1988604c..c3889012b 100644 --- a/xwords4/common/game.c +++ b/xwords4/common/game.c @@ -49,7 +49,8 @@ static void gi_setDict( MPFORMAL CurGameInfo* gi, const DictionaryCtxt* dict ); #endif static void -checkServerRole( CurGameInfo* gi, XP_U16* nPlayersHere, XP_U16* nPlayersTotal ) +checkServerRole( CurGameInfo* gi, XP_U16* nPlayersHere, + XP_U16* nPlayersTotal ) { if ( !!gi ) { XP_U16 ii, remoteCount = 0; @@ -87,7 +88,7 @@ makeGameID( XW_UtilCtxt* util ) void game_makeNewGame( MPFORMAL XWGame* game, CurGameInfo* gi, XW_UtilCtxt* util, DrawCtx* draw, - CommonPrefs* cp, const TransportProcs* procs + const CommonPrefs* cp, const TransportProcs* procs #ifdef SET_GAMESEED ,XP_U16 gameSeed #endif @@ -280,6 +281,25 @@ game_makeFromStream( MPFORMAL XWStreamCtxt* stream, XWGame* game, return success; } /* game_makeFromStream */ +void +game_saveNewGame( MPFORMAL const CurGameInfo* gi, XW_UtilCtxt* util, + const CommonPrefs* cp, XWStreamCtxt* out ) +{ + XWGame newGame = {0}; + CurGameInfo newGI = {0}; + gi_copy( MPPARM(mpool) &newGI, gi ); + + game_makeNewGame( MPPARM(mpool) &newGame, &newGI, util, + NULL, /* DrawCtx*, */ + cp, NULL, /* TransportProcs* procs */ + 0 ); + + game_saveToStream( &newGame, &newGI, out, 1 ); + game_saveSucceeded( &newGame, 1 ); + game_dispose( &newGame ); + gi_disposePlayerInfo( MPPARM(mpool) &newGI ); +} + void game_saveToStream( const XWGame* game, const CurGameInfo* gi, XWStreamCtxt* stream, XP_U16 saveToken ) diff --git a/xwords4/common/game.h b/xwords4/common/game.h index 857f70daa..b2d0cc070 100644 --- a/xwords4/common/game.h +++ b/xwords4/common/game.h @@ -60,7 +60,7 @@ typedef struct XWGame { void game_makeNewGame( MPFORMAL XWGame* game, CurGameInfo* gi, XW_UtilCtxt* util, DrawCtx* draw, - CommonPrefs* cp, const TransportProcs* procs + const CommonPrefs* cp, const TransportProcs* procs #ifdef SET_GAMESEED ,XP_U16 gameSeed #endif @@ -76,6 +76,9 @@ XP_Bool game_makeFromStream( MPFORMAL XWStreamCtxt* stream, XWGame* game, DrawCtx* draw, CommonPrefs* cp, const TransportProcs* procs ); +void game_saveNewGame( MPFORMAL const CurGameInfo* gi, XW_UtilCtxt* util, + const CommonPrefs* cp, XWStreamCtxt* out ); + void game_saveToStream( const XWGame* game, const CurGameInfo* gi, XWStreamCtxt* stream, XP_U16 saveToken ); void game_saveSucceeded( const XWGame* game, XP_U16 saveToken ); diff --git a/xwords4/common/invit.c b/xwords4/common/invit.c index 0d0c34b6f..494e3952b 100644 --- a/xwords4/common/invit.c +++ b/xwords4/common/invit.c @@ -55,6 +55,7 @@ void invit_setDevID( InviteInfo* invit, XP_U32 devID ) { invit->devID = devID; + types_addType( &invit->_conTypes, COMMS_CONN_RELAY ); } void diff --git a/xwords4/common/invit.h b/xwords4/common/invit.h index def003f28..3f160bf31 100644 --- a/xwords4/common/invit.h +++ b/xwords4/common/invit.h @@ -30,6 +30,12 @@ typedef enum {OSType_NONE, OSType_LINUX, OSType_ANDROID, } XP_OSType; +/* InviteInfo + * + * A representation of return addresses sent with an invitation so that the + * recipient has all it needs to create a game and connect back. + */ + typedef struct _InviteInfo { XP_U8 version; /* struct version for backward compatibility */ XP_U16 _conTypes; diff --git a/xwords4/linux/gamesdb.c b/xwords4/linux/gamesdb.c index c49e3c533..e3fa4f2c0 100644 --- a/xwords4/linux/gamesdb.c +++ b/xwords4/linux/gamesdb.c @@ -25,6 +25,7 @@ static void getColumnText( sqlite3_stmt *ppStmt, int iCol, XP_UCHAR* buf, int len ); #ifdef DEBUG static char* sqliteErr2str( int err ); +static void assertPrintResult( sqlite3* pDb, int result, int expect ); #endif sqlite3* @@ -41,6 +42,7 @@ openGamesDB( const char* dbName ) "CREATE TABLE games ( " "rowid INTEGER PRIMARY KEY AUTOINCREMENT" ",game BLOB" + ",inviteInfo BLOB" ",room VARCHAR(32)" ",connvia VARCHAR(32)" ",ended INT(1)" @@ -48,6 +50,7 @@ openGamesDB( const char* dbName ) ",nmoves INT" ",seed INT" ",gameid INT" + ",ntotal INT(2)" ",nmissing INT(2)" ")"; result = sqlite3_exec( pDb, createGamesStr, NULL, NULL, NULL ); @@ -68,31 +71,32 @@ closeGamesDB( sqlite3* pDb ) XP_LOGF( "%s finished", __func__ ); } -void -writeToDB( XWStreamCtxt* stream, void* closure ) +static sqlite3_int64 +writeBlobColumn( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 curRow, + const char* column ) { int result; - CommonGlobals* cGlobals = (CommonGlobals*)closure; - sqlite3_int64 selRow = cGlobals->selRow; - sqlite3* pDb = cGlobals->pDb; + /* size includes stream version as header */ XP_U16 len = stream_getSize( stream ); char buf[256]; char* query; sqlite3_stmt* stmt = NULL; - XP_Bool newGame = -1 == selRow; + XP_Bool newGame = -1 == curRow; if ( newGame ) { /* new row; need to insert blob first */ - query = "INSERT INTO games (game) VALUES (?)"; + const char* fmt = "INSERT INTO games (%s) VALUES (?)"; + snprintf( buf, sizeof(buf), fmt, column ); + query = buf; } else { - const char* fmt = "UPDATE games SET game=? where rowid=%lld"; - snprintf( buf, sizeof(buf), fmt, selRow ); + const char* fmt = "UPDATE games SET %s=? where rowid=%lld"; + snprintf( buf, sizeof(buf), fmt, column, curRow ); query = buf; } - result = sqlite3_prepare_v2( pDb, query, -1, &stmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); - result = sqlite3_bind_zeroblob( stmt, 1 /*col 0 ??*/, len ); - XP_ASSERT( SQLITE_OK == result ); + result = sqlite3_prepare_v2( pDb, query, -1, &stmt, NULL ); + assertPrintResult( pDb, result, SQLITE_OK ); + result = sqlite3_bind_zeroblob( stmt, 1 /*col 0 ??*/, sizeof(XP_U16) + len ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( stmt ); if ( SQLITE_DONE != result ) { XP_LOGF( "%s: sqlite3_step => %s", __func__, sqliteErr2str( result ) ); @@ -101,24 +105,51 @@ writeToDB( XWStreamCtxt* stream, void* closure ) XP_USE( result ); if ( newGame ) { /* new row; need to insert blob first */ - selRow = sqlite3_last_insert_rowid( pDb ); - XP_LOGF( "%s: new rowid: %lld", __func__, selRow ); - cGlobals->selRow = selRow; + curRow = sqlite3_last_insert_rowid( pDb ); + XP_LOGF( "%s: new rowid: %lld", __func__, curRow ); } sqlite3_blob* blob; - result = sqlite3_blob_open( pDb, "main", "games", "game", - selRow, 1 /*flags: writeable*/, &blob ); - XP_ASSERT( SQLITE_OK == result ); + result = sqlite3_blob_open( pDb, "main", "games", column, + curRow, 1 /*flags: writeable*/, &blob ); + assertPrintResult( pDb, result, SQLITE_OK ); + XP_U16 strVersion = stream_getVersion( stream ); + XP_ASSERT( strVersion <= CUR_STREAM_VERS ); + result = sqlite3_blob_write( blob, &strVersion, sizeof(strVersion), 0 ); + assertPrintResult( pDb, result, SQLITE_OK ); const XP_U8* ptr = stream_getPtr( stream ); - result = sqlite3_blob_write( blob, ptr, len, 0 ); - XP_ASSERT( SQLITE_OK == result ); + result = sqlite3_blob_write( blob, ptr, len, sizeof(strVersion) ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_blob_close( blob ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); if ( !!stmt ) { sqlite3_finalize( stmt ); } + return curRow; +} + +sqlite3_int64 +writeNewGameToDB( XWStreamCtxt* stream, sqlite3* pDb ) +{ + sqlite3_int64 newRow = writeBlobColumn( stream, pDb, -1, "game" ); + return newRow; +} + +void +writeToDB( XWStreamCtxt* stream, void* closure ) +{ + CommonGlobals* cGlobals = (CommonGlobals*)closure; + sqlite3_int64 selRow = cGlobals->selRow; + sqlite3* pDb = cGlobals->pDb; + + XP_Bool newGame = -1 == selRow; + selRow = writeBlobColumn( stream, pDb, selRow, "game" ); + + if ( newGame ) { /* new row; need to insert blob first */ + cGlobals->selRow = selRow; + } + (*cGlobals->onSave)( cGlobals->onSaveClosure, selRow, newGame ); } @@ -130,6 +161,7 @@ summarize( CommonGlobals* cGlobals ) XP_S16 turn = server_getCurrentTurn( cGlobals->game.server ); XP_U16 seed = 0; XP_S16 nMissing = 0; + XP_U16 nTotal = cGlobals->gi->nPlayers; XP_U32 gameID = cGlobals->gi->gameID; XP_ASSERT( 0 != gameID ); CommsAddrRec addr = {0}; @@ -171,15 +203,15 @@ summarize( CommonGlobals* cGlobals ) } const char* fmt = "UPDATE games " - " SET room='%s', ended=%d, turn=%d, nmissing=%d, nmoves=%d, seed=%d, gameid=%d, connvia='%s'" + " SET room='%s', ended=%d, turn=%d, ntotal=%d, nmissing=%d, nmoves=%d, seed=%d, gameid=%d, connvia='%s'" " WHERE rowid=%lld"; XP_UCHAR buf[256]; - snprintf( buf, sizeof(buf), fmt, room, gameOver?1:0, turn, nMissing, nMoves, - seed, gameID, connvia, cGlobals->selRow ); + snprintf( buf, sizeof(buf), fmt, room, gameOver?1:0, turn, nTotal, nMissing, + nMoves, seed, gameID, connvia, cGlobals->selRow ); XP_LOGF( "query: %s", buf ); sqlite3_stmt* stmt = NULL; int result = sqlite3_prepare_v2( cGlobals->pDb, buf, -1, &stmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( cGlobals->pDb, result, SQLITE_OK ); result = sqlite3_step( stmt ); if ( SQLITE_DONE != result ) { XP_LOGF( "sqlite3_step=>%s", sqliteErr2str( result ) ); @@ -198,7 +230,7 @@ listGames( sqlite3* pDb ) int result = sqlite3_prepare_v2( pDb, "SELECT rowid FROM games ORDER BY rowid", -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); XP_USE( result ); while ( NULL != ppStmt ) { switch( sqlite3_step( ppStmt ) ) { @@ -226,14 +258,15 @@ XP_Bool getGameInfo( sqlite3* pDb, sqlite3_int64 rowid, GameInfo* gib ) { XP_Bool success = XP_FALSE; - const char* fmt = "SELECT room, ended, turn, nmoves, nmissing, seed, connvia, gameid " + const char* fmt = "SELECT room, ended, turn, nmoves, ntotal, nmissing, " + "seed, connvia, gameid " "FROM games WHERE rowid = %lld"; XP_UCHAR query[256]; snprintf( query, sizeof(query), fmt, rowid ); sqlite3_stmt* ppStmt; int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( ppStmt ); if ( SQLITE_ROW == result ) { success = XP_TRUE; @@ -241,10 +274,11 @@ getGameInfo( sqlite3* pDb, sqlite3_int64 rowid, GameInfo* gib ) gib->gameOver = 1 == sqlite3_column_int( ppStmt, 1 ); gib->turn = sqlite3_column_int( ppStmt, 2 ); gib->nMoves = sqlite3_column_int( ppStmt, 3 ); - gib->nMissing = sqlite3_column_int( ppStmt, 4 ); - gib->seed = sqlite3_column_int( ppStmt, 5 ); - getColumnText( ppStmt, 6, gib->conn, sizeof(gib->conn) ); - gib->gameID = sqlite3_column_int( ppStmt, 7 ); + gib->nTotal = sqlite3_column_int( ppStmt, 4 ); + gib->nMissing = sqlite3_column_int( ppStmt, 5 ); + gib->seed = sqlite3_column_int( ppStmt, 6 ); + getColumnText( ppStmt, 7, gib->conn, sizeof(gib->conn) ); + gib->gameID = sqlite3_column_int( ppStmt, 8 ); snprintf( gib->name, sizeof(gib->name), "Game %lld", rowid ); } sqlite3_finalize( ppStmt ); @@ -259,11 +293,11 @@ getRowsForGameID( sqlite3* pDb, XP_U32 gameID, sqlite3_int64* rowids, *nRowIDs = 0; char buf[256]; - snprintf( buf, sizeof(buf), "SELECT rowid from games WHERE gameid = %d LIMIT %d", - gameID, maxRowIDs ); + snprintf( buf, sizeof(buf), "SELECT rowid from games WHERE gameid = %d " + "LIMIT %d", gameID, maxRowIDs ); sqlite3_stmt *ppStmt; int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); int ii; for ( ii = 0; ii < maxRowIDs; ++ii ) { result = sqlite3_step( ppStmt ); @@ -276,26 +310,56 @@ getRowsForGameID( sqlite3* pDb, XP_U32 gameID, sqlite3_int64* rowids, sqlite3_finalize( ppStmt ); } -XP_Bool -loadGame( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid ) +static XP_Bool +loadBlobColumn( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid, + const char* column ) { char buf[256]; - snprintf( buf, sizeof(buf), "SELECT game from games WHERE rowid = %lld", rowid ); + snprintf( buf, sizeof(buf), "SELECT %s from games WHERE rowid = %lld", + column, rowid ); sqlite3_stmt *ppStmt; int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( ppStmt ); XP_Bool success = SQLITE_ROW == result; if ( success ) { const void* ptr = sqlite3_column_blob( ppStmt, 0 ); int size = sqlite3_column_bytes( ppStmt, 0 ); - stream_putBytes( stream, ptr, size ); + success = 0 < size; + if ( success ) { + XP_U16 strVersion; + XP_MEMCPY( &strVersion, ptr, sizeof(strVersion) ); + XP_ASSERT( strVersion <= CUR_STREAM_VERS ); + stream_setVersion( stream, strVersion ); + XP_ASSERT( size >= sizeof(strVersion) ); + stream_putBytes( stream, ptr + sizeof(strVersion), + size - sizeof(strVersion) ); + } } sqlite3_finalize( ppStmt ); return success; } +XP_Bool +loadGame( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid ) +{ + return loadBlobColumn( stream, pDb, rowid, "game" ); +} + +void +saveInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid ) +{ + sqlite3_int64 row = writeBlobColumn( stream, pDb, rowid, "inviteInfo" ); + assert( row == rowid ); +} + +XP_Bool +loadInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid ) +{ + return loadBlobColumn( stream, pDb, rowid, "inviteInfo" ); +} + void deleteGame( sqlite3* pDb, sqlite3_int64 rowid ) { @@ -303,9 +367,9 @@ deleteGame( sqlite3* pDb, sqlite3_int64 rowid ) snprintf( query, sizeof(query), "DELETE FROM games WHERE rowid = %lld", rowid ); sqlite3_stmt* ppStmt; int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( ppStmt ); - XP_ASSERT( SQLITE_DONE == result ); + assertPrintResult( pDb, result, SQLITE_DONE ); XP_USE( result ); sqlite3_finalize( ppStmt ); } @@ -319,9 +383,9 @@ db_store( sqlite3* pDb, const gchar* key, const gchar* value ) key, value ); sqlite3_stmt *ppStmt; int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( ppStmt ); - XP_ASSERT( SQLITE_DONE == result ); + assertPrintResult( pDb, result, SQLITE_DONE ); XP_USE( result ); sqlite3_finalize( ppStmt ); } @@ -355,9 +419,9 @@ db_remove( sqlite3* pDb, const gchar* key ) snprintf( query, sizeof(query), "DELETE FROM pairs WHERE key = '%s'", key ); sqlite3_stmt *ppStmt; int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL ); - XP_ASSERT( SQLITE_OK == result ); + assertPrintResult( pDb, result, SQLITE_OK ); result = sqlite3_step( ppStmt ); - XP_ASSERT( SQLITE_DONE == result ); + assertPrintResult( pDb, result, SQLITE_DONE ); XP_USE( result ); sqlite3_finalize( ppStmt ); } @@ -413,4 +477,17 @@ sqliteErr2str( int err ) } return ""; } + +static void +assertPrintResult( sqlite3* pDb, int result, int expect ) +{ + int code = sqlite3_errcode( pDb ); + XP_ASSERT( code == result ); /* do I need to pass it? */ + if ( code != expect ) { + const char* msg = sqlite3_errmsg( pDb ); + XP_LOGF( "sqlite3 error: %s", msg ); + XP_ASSERT(0); + } +} + #endif diff --git a/xwords4/linux/gamesdb.h b/xwords4/linux/gamesdb.h index 4a8bf9a76..b9056e2ed 100644 --- a/xwords4/linux/gamesdb.h +++ b/xwords4/linux/gamesdb.h @@ -35,6 +35,7 @@ typedef struct _GameInfo { XP_S16 nMoves; XP_Bool gameOver; XP_S16 turn; + XP_U16 nTotal; XP_S16 nMissing; XP_U16 seed; } GameInfo; @@ -44,6 +45,8 @@ sqlite3* openGamesDB( const char* dbName ); void closeGamesDB( sqlite3* dbp ); void writeToDB( XWStreamCtxt* stream, void* closure ); +sqlite3_int64 writeNewGameToDB( XWStreamCtxt* stream, sqlite3* pDb ); + void summarize( CommonGlobals* cGlobals ); /* Return GSList whose data is (ptrs to) rowids */ @@ -52,6 +55,10 @@ XP_Bool getGameInfo( sqlite3* dbp, sqlite3_int64 rowid, GameInfo* gib ); void getRowsForGameID( sqlite3* dbp, XP_U32 gameID, sqlite3_int64* rowids, int* nRowIDs ); XP_Bool loadGame( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid ); +void saveInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, + sqlite3_int64 rowid ); +XP_Bool loadInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, + sqlite3_int64 rowid ); void deleteGame( sqlite3* pDb, sqlite3_int64 rowid ); #define KEY_RDEVID "RDEVID" diff --git a/xwords4/linux/gtkaskm.c b/xwords4/linux/gtkaskm.c index aef7ef644..3560c569b 100644 --- a/xwords4/linux/gtkaskm.c +++ b/xwords4/linux/gtkaskm.c @@ -77,8 +77,8 @@ gtkaskm( const gchar* message, AskMInfo* infos, int nInfos ) for ( ii = 0; ii < nInfos; ++ii ) { AskMInfo* info = &infos[ii]; if ( !state.cancelled ) { - const gchar* txt = gtk_entry_get_text( GTK_ENTRY(fields[ii]) ); - XP_LOGF( "%s: got text %s", __func__, txt ); + XP_LOGF( "%s: got text %s", __func__, + gtk_entry_get_text( GTK_ENTRY(fields[ii]) ) ); } else { *info->result = NULL; } diff --git a/xwords4/linux/gtkboard.c b/xwords4/linux/gtkboard.c index 440e15be8..d2a19a0b9 100644 --- a/xwords4/linux/gtkboard.c +++ b/xwords4/linux/gtkboard.c @@ -54,6 +54,7 @@ #include "draw.h" #include "game.h" #include "movestak.h" +#include "strutils.h" #include "gtkask.h" #include "gtkinvit.h" #include "gtkaskm.h" @@ -82,6 +83,10 @@ static GtkWidget* addButton( GtkWidget* hbox, gchar* label, GCallback func, static void handle_invite_button( GtkWidget* widget, GtkGameGlobals* globals ); static void gtkShowFinalScores( const GtkGameGlobals* globals, XP_Bool ignoreTimeout ); +static void send_invites( CommonGlobals* cGlobals, + const CommsAddrRec* inviteAddr, + const XP_UCHAR* relayID, + XP_U16 nPlayers ); #define GTK_TRAY_HT_ROWS 3 @@ -103,7 +108,7 @@ lookupClientStream( GtkGameGlobals* globals, int sock ) static void rememberClient( GtkGameGlobals* globals, guint key, int sock, - XWStreamCtxt* stream ) + XWStreamCtxt* stream ) { short i; for ( i = 0; i < MAX_NUM_PLAYERS; ++i ) { @@ -868,10 +873,39 @@ destroy_board_window( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) static void on_board_window_shown( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) { - if ( server_getGameIsOver( globals->cGlobals.game.server ) ) { + LOG_FUNC(); + CommonGlobals* cGlobals = &globals->cGlobals; + if ( server_getGameIsOver( cGlobals->game.server ) ) { gtkShowFinalScores( globals, XP_TRUE ); } -} + + CommsCtxt* comms = cGlobals->game.comms; + if ( !!comms /*&& COMMS_CONN_NONE == comms_getConTypes( comms )*/ ) { + /* If it has pending invite info, send the invitation! */ + XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) + cGlobals->params->vtMgr, + cGlobals, CHANNEL_NONE, NULL ); + if ( loadInviteAddrs( stream, cGlobals->pDb, cGlobals->selRow ) ) { + CommsAddrRec addr = {0}; + addrFromStream( &addr, stream ); + comms_setAddr( cGlobals->game.comms, &addr ); + + XP_U16 nRecs = stream_getU8( stream ); + XP_LOGF( "%s: got invite info: %d records", __func__, nRecs ); + for ( int ii = 0; ii < nRecs; ++ii ) { + XP_UCHAR relayID[32]; + stringFromStreamHere( stream, relayID, sizeof(relayID) ); + XP_LOGF( "%s: loaded relayID %s", __func__, relayID ); + + CommsAddrRec addr = {0}; + addrFromStream( &addr, stream ); + + send_invites( cGlobals, &addr, relayID, 1 /*nPlayers*/ ); + } + } + stream_destroy( stream ); + } +} /* on_board_window_shown */ static void cleanup( GtkGameGlobals* globals ) @@ -986,7 +1020,9 @@ new_game_impl( GtkGameGlobals* globals, XP_Bool fireConnDlg ) if ( !!cGlobals->game.comms ) { comms_getAddr( cGlobals->game.comms, &addr ); } else { - comms_getInitialAddr( &addr, RELAY_NAME_DEFAULT, RELAY_PORT_DEFAULT ); + XP_U32 devID = linux_getDevIDRelay( cGlobals->params ); + comms_getInitialAddr( &addr, RELAY_NAME_DEFAULT, RELAY_PORT_DEFAULT, + devID ); } CurGameInfo* gi = cGlobals->gi; @@ -1572,7 +1608,7 @@ static void handle_invite_button( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) { CommonGlobals* cGlobals = &globals->cGlobals; - const CurGameInfo* gi = cGlobals->gi; + /* const CurGameInfo* gi = cGlobals->gi; */ /* gchar* countStr; */ /* gchar* phone = NULL; */ @@ -1595,46 +1631,52 @@ handle_invite_button( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) CommsAddrRec inviteAddr = {0}; gint nPlayers = nMissing; - XP_U32 devID; - XP_Bool confirmed = gtkInviteDlg( globals, &inviteAddr, &nPlayers, &devID ); + XP_Bool confirmed = gtkInviteDlg( globals, &inviteAddr, &nPlayers ); XP_LOGF( "%s: inviteDlg => %d", __func__, confirmed ); if ( confirmed ) { - gchar gameName[64]; - snprintf( gameName, VSIZE(gameName), "Game %d", gi->gameID ); + send_invites( cGlobals, &inviteAddr, NULL, nPlayers ); + } +} /* handle_invite_button */ - CommsAddrRec addr; - CommsCtxt* comms = cGlobals->game.comms; - XP_ASSERT( comms ); - comms_getAddr( comms, &addr ); +static void +send_invites( CommonGlobals* cGlobals, const CommsAddrRec* inviteAddr, + const XP_UCHAR* relayID, XP_U16 nPlayers ) +{ + CommsAddrRec addr = {0}; + CommsCtxt* comms = cGlobals->game.comms; + XP_ASSERT( comms ); + comms_getAddr( comms, &addr ); - gint forceChannel = 0; /* PENDING */ + gint forceChannel = 0; /* PENDING */ - InviteInfo invit = {0}; - invit_init( &invit, gi, &addr, nPlayers, forceChannel ); - invit_setDevID( &invit, linux_getDevIDRelay( cGlobals->params ) ); + InviteInfo invit = {0}; + invit_init( &invit, cGlobals->gi, &addr, nPlayers, forceChannel ); + invit_setDevID( &invit, linux_getDevIDRelay( cGlobals->params ) ); #ifdef DEBUG - { - XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) - cGlobals->params->vtMgr, - NULL, CHANNEL_NONE, NULL ); - invit_saveToStream( &invit, stream ); - InviteInfo tmp; - invit_makeFromStream( &tmp, stream ); - stream_destroy( stream ); - XP_ASSERT( 0 == memcmp( &invit, &tmp, sizeof(invit) ) ); - } + { + XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) + cGlobals->params->vtMgr, + NULL, CHANNEL_NONE, NULL ); + invit_saveToStream( &invit, stream ); + InviteInfo tmp; + invit_makeFromStream( &tmp, stream ); + stream_destroy( stream ); + XP_ASSERT( 0 == memcmp( &invit, &tmp, sizeof(invit) ) ); + } #endif - if ( addr_hasType( &inviteAddr, COMMS_CONN_SMS ) ) { - linux_sms_invite( cGlobals->params, gi, &addr, gameName, - nPlayers, forceChannel, - inviteAddr.u.sms.phone, inviteAddr.u.sms.port ); - } - if ( addr_hasType( &addr, COMMS_CONN_RELAY ) ) { - relaycon_invite( cGlobals->params, devID, &invit ); - } + if ( addr_hasType( inviteAddr, COMMS_CONN_SMS ) ) { + XP_ASSERT( 0 ); /* not implemented */ + /* linux_sms_invite( cGlobals->params, gi, &addr, gameName, */ + /* nPlayers, forceChannel, */ + /* inviteAddr.u.sms.phone, inviteAddr.u.sms.port ); */ + } + if ( addr_hasType( inviteAddr, COMMS_CONN_RELAY ) ) { + XP_U32 devID = inviteAddr->u.ip_relay.devID; + XP_ASSERT( 0 != devID || (!!relayID && !!relayID[0]) ); + relaycon_invite( cGlobals->params, devID, relayID, &invit ); } /* while ( gtkaskm( "Invite how many and how?", infos, VSIZE(infos) ) ) { */ @@ -1673,7 +1715,7 @@ handle_invite_button( GtkWidget* XP_UNUSED(widget), GtkGameGlobals* globals ) /* for ( int ii = 0; ii < VSIZE(infos); ++ii ) { */ /* g_free( *infos[ii].result ); */ /* } */ -} /* handle_invite_button */ +} static void gtkUserError( GtkGameGlobals* globals, const char* format, ... ) @@ -1805,8 +1847,7 @@ gtkShowFinalScores( const GtkGameGlobals* globals, XP_Bool ignoreTimeout ) buttons, timeout ); free( text ); if ( 2 == chosen ) { - XP_LOGF( "%s: rematch chosen!", __func__ ); - XP_ASSERT( 0 ); + make_rematch( globals->apg, cGlobals ); } } /* gtkShowFinalScores */ @@ -1847,7 +1888,7 @@ gtk_util_notifyGameOver( XW_UtilCtxt* uc, XP_S16 quitter ) server_handleUndo( cGlobals->game.server, 0 ); board_draw( cGlobals->game.board ); } else if ( !cGlobals->params->skipGameOver ) { - gtkShowFinalScores( globals, XP_FALSE ); + gtkShowFinalScores( globals, XP_TRUE ); } } /* gtk_util_notifyGameOver */ @@ -2785,7 +2826,8 @@ makeNewGame( GtkGameGlobals* globals ) if ( 0 == relayPort ) { relayPort = RELAY_PORT_DEFAULT; } - comms_getInitialAddr( &cGlobals->addr, relayName, relayPort ); + comms_getInitialAddr( &cGlobals->addr, relayName, relayPort, + linux_getDevIDRelay( cGlobals->params ) ); } CurGameInfo* gi = cGlobals->gi; diff --git a/xwords4/linux/gtkinvit.c b/xwords4/linux/gtkinvit.c index 2aa9d5cea..180473e38 100644 --- a/xwords4/linux/gtkinvit.c +++ b/xwords4/linux/gtkinvit.c @@ -36,7 +36,6 @@ typedef struct _GtkInviteState { GtkGameGlobals* globals; CommsAddrRec* addr; gint* nPlayersP; - XP_U32* devIDP; gint maxPlayers; XP_UCHAR devIDBuf[32]; @@ -94,7 +93,7 @@ handle_ok( GtkWidget* XP_UNUSED(widget), gpointer closure ) #ifdef XWFEATURE_RELAY case COMMS_CONN_RELAY: txt = gtk_entry_get_text( GTK_ENTRY(state->devID) ); - *(state->devIDP) = atoi( txt ); + state->addr->u.ip_relay.devID = atoi( txt ); break; #endif #ifdef XWFEATURE_BLUETOOTH @@ -276,17 +275,15 @@ onPageChanged( GtkNotebook* XP_UNUSED(notebook), gpointer XP_UNUSED(arg1), } XP_Bool -gtkInviteDlg( GtkGameGlobals* globals, CommsAddrRec* addr, gint* nPlayers, - XP_U32* devIDP ) +gtkInviteDlg( GtkGameGlobals* globals, CommsAddrRec* addr, gint* nPlayersP ) { GtkInviteState state; XP_MEMSET( &state, 0, sizeof(state) ); state.globals = globals; state.addr = addr; - state.nPlayersP = nPlayers; - state.maxPlayers = *nPlayers; - state.devIDP = devIDP; + state.nPlayersP = nPlayersP; + state.maxPlayers = *nPlayersP; GtkWidget* dialog; GtkWidget* hbox; diff --git a/xwords4/linux/gtkinvit.h b/xwords4/linux/gtkinvit.h index ba2b34596..de97d81b4 100644 --- a/xwords4/linux/gtkinvit.h +++ b/xwords4/linux/gtkinvit.h @@ -24,7 +24,7 @@ /* return true if not cancelled */ XP_Bool gtkInviteDlg( GtkGameGlobals* globals, CommsAddrRec* addr, - /*inout*/ gint* nPlayers, XP_U32* devID ); + /*inout*/ gint* nPlayers ); #endif diff --git a/xwords4/linux/gtkmain.c b/xwords4/linux/gtkmain.c index c0fc65c9f..d4d5e8a8e 100644 --- a/xwords4/linux/gtkmain.c +++ b/xwords4/linux/gtkmain.c @@ -33,7 +33,6 @@ static void onNewData( GtkAppGlobals* apg, sqlite3_int64 rowid, XP_Bool isNew ); static void updateButtons( GtkAppGlobals* apg ); -static void open_row( GtkAppGlobals* apg, sqlite3_int64 row ); static void recordOpened( GtkAppGlobals* apg, GtkGameGlobals* globals ) @@ -77,7 +76,7 @@ findOpenGame( const GtkAppGlobals* apg, sqlite3_int64 rowid ) } enum { ROW_ITEM, NAME_ITEM, ROOM_ITEM, GAMEID_ITEM, SEED_ITEM, CONN_ITEM, OVER_ITEM, TURN_ITEM, - NMOVES_ITEM, MISSING_ITEM, N_ITEMS }; + NMOVES_ITEM, NTOTAL_ITEM, MISSING_ITEM, N_ITEMS }; static void foreachProc( GtkTreeModel* model, GtkTreePath* XP_UNUSED(path), @@ -111,7 +110,7 @@ row_activated_cb( GtkTreeView* tree_view, GtkTreePath* path, if ( gtk_tree_model_get_iter( model, &iter, path ) ) { sqlite3_int64 rowid; gtk_tree_model_get( model, &iter, ROW_ITEM, &rowid, -1 ); - open_row( apg, rowid ); + open_row( apg, rowid, XP_FALSE ); } } @@ -158,6 +157,7 @@ init_games_list( GtkAppGlobals* apg ) addTextColumn( list, "Ended", OVER_ITEM ); addTextColumn( list, "Turn", TURN_ITEM ); addTextColumn( list, "NMoves", NMOVES_ITEM ); + addTextColumn( list, "NTotal", NTOTAL_ITEM ); addTextColumn( list, "NMissing", MISSING_ITEM ); GtkListStore* store = gtk_list_store_new( N_ITEMS, @@ -170,6 +170,7 @@ init_games_list( GtkAppGlobals* apg ) G_TYPE_BOOLEAN, /* OVER_ITEM */ G_TYPE_INT, /* TURN_ITEM */ G_TYPE_INT, /* NMOVES_ITEM */ + G_TYPE_INT, /* NTOTAL_ITEM */ G_TYPE_INT /* MISSING_ITEM */ ); gtk_tree_view_set_model( GTK_TREE_VIEW(list), GTK_TREE_MODEL(store) ); @@ -217,6 +218,7 @@ add_to_list( GtkWidget* list, sqlite3_int64 rowid, XP_Bool isNew, OVER_ITEM, gib->gameOver, TURN_ITEM, gib->turn, NMOVES_ITEM, gib->nMoves, + NTOTAL_ITEM, gib->nTotal, MISSING_ITEM, gib->nMissing, -1 ); XP_LOGF( "DONE adding" ); @@ -226,7 +228,8 @@ static void updateButtons( GtkAppGlobals* apg ) { guint count = apg->selRows->len; - gtk_widget_set_sensitive( apg->openButton, 1 == count ); + gtk_widget_set_sensitive( apg->openButton, 1 <= count ); + gtk_widget_set_sensitive( apg->rematchButton, 1 == count ); gtk_widget_set_sensitive( apg->deleteButton, 1 <= count ); } @@ -249,10 +252,14 @@ handle_newgame_button( GtkWidget* XP_UNUSED(widget), void* closure ) } } -static void -open_row( GtkAppGlobals* apg, sqlite3_int64 row ) +void +open_row( GtkAppGlobals* apg, sqlite3_int64 row, XP_Bool isNew ) { if ( -1 != row && !gameIsOpen( apg, row ) ) { + if ( isNew ) { + onNewData( apg, row, XP_TRUE ); + } + apg->params->needsNewGame = XP_FALSE; GtkGameGlobals* globals = malloc( sizeof(*globals) ); initGlobals( globals, apg->params, NULL ); @@ -267,8 +274,82 @@ static void handle_open_button( GtkWidget* XP_UNUSED(widget), void* closure ) { GtkAppGlobals* apg = (GtkAppGlobals*)closure; - sqlite3_int64 selRow = getSelRow( apg ); - open_row( apg, selRow ); + + GArray* selRows = apg->selRows; + for ( int ii = 0; ii < selRows->len; ++ii ) { + sqlite3_int64 row = g_array_index( selRows, sqlite3_int64, ii ); + open_row( apg, row, XP_FALSE ); + } +} + +void +make_rematch( GtkAppGlobals* apg, const CommonGlobals* cGlobals ) +{ + // LaunchParams* params = apg->params; + XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) + cGlobals->params->vtMgr, + NULL, CHANNEL_NONE, NULL ); + + /* Create new game. But has no addressing info, so need to set that + aside for later. */ + CurGameInfo gi = {0}; + gi_copy( MPPARM(cGlobals->util->mpool) &gi, cGlobals->gi ); + gi.gameID = 0; /* clear so will get generated */ + game_saveNewGame( MPPARM(cGlobals->util->mpool) &gi, + cGlobals->util, &cGlobals->cp, stream ); + + sqlite3_int64 rowID = writeNewGameToDB( stream, cGlobals->pDb ); + stream_destroy( stream ); + gi_disposePlayerInfo( MPPARM(cGlobals->util->mpool) &gi ); + + /* If it's a multi-device game, save enough information with it than when + opened it can invite the other device[s] join the rematch. */ + const CommsCtxt* comms = cGlobals->game.comms; + if ( !!comms ) { + XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) + cGlobals->params->vtMgr, + NULL, CHANNEL_NONE, NULL ); + CommsAddrRec addr; + comms_getAddr( comms, &addr ); + addrToStream( stream, &addr ); + + CommsAddrRec addrs[4]; + XP_U16 nRecs = VSIZE(addrs); + comms_getAddrs( comms, addrs, &nRecs ); + + stream_putU8( stream, nRecs ); + for ( int ii = 0; ii < nRecs; ++ii ) { + XP_UCHAR relayID[32]; + XP_U16 len = sizeof(relayID); + comms_formatRelayID( comms, ii, relayID, &len ); + XP_LOGF( "%s: adding relayID: %s", __func__, relayID ); + stringToStream( stream, relayID ); + if ( addr_hasType( &addrs[ii], COMMS_CONN_RELAY ) ) { + /* copy over room name */ + XP_STRCAT( addrs[ii].u.ip_relay.invite, addr.u.ip_relay.invite ); + } + addrToStream( stream, &addrs[ii] ); + } + saveInviteAddrs( stream, cGlobals->pDb, rowID ); + stream_destroy( stream ); + } + + open_row( apg, rowID, XP_TRUE ); +} + +static void +handle_rematch_button( GtkWidget* XP_UNUSED(widget), void* closure ) +{ + GtkAppGlobals* apg = (GtkAppGlobals*)closure; + GArray* selRows = apg->selRows; + for ( int ii = 0; ii < selRows->len; ++ii ) { + sqlite3_int64 rowid = g_array_index( selRows, sqlite3_int64, ii ); + GtkGameGlobals tmpGlobals; + if ( loadGameNoDraw( &tmpGlobals, apg->params, rowid ) ) { + make_rematch( apg, &tmpGlobals.cGlobals ); + } + freeGlobals( &tmpGlobals ); + } } static void @@ -396,6 +477,8 @@ makeGamesWindow( GtkAppGlobals* apg ) (void)addButton( "New game", hbox, G_CALLBACK(handle_newgame_button), apg ); apg->openButton = addButton( "Open", hbox, G_CALLBACK(handle_open_button), apg ); + apg->rematchButton = addButton( "Rematch", hbox, + G_CALLBACK(handle_rematch_button), apg ); apg->deleteButton = addButton( "Delete", hbox, G_CALLBACK(handle_delete_button), apg ); (void)addButton( "Quit", hbox, G_CALLBACK(handle_quit_button), apg ); @@ -479,29 +562,38 @@ relayInviteReceived( void* closure, InviteInfo* invite ) GtkAppGlobals* apg = (GtkAppGlobals*)closure; LaunchParams* params = apg->params; - CurGameInfo gi = {0}; - gi_copy( MPPARM(params->mpool) &gi, ¶ms->pgi ); + XP_U32 gameID = invite->gameID; + sqlite3_int64 rowids[1]; + int nRowIDs = VSIZE(rowids); + getRowsForGameID( apg->params->pDb, gameID, rowids, &nRowIDs ); + + if ( 0 < nRowIDs ) { + gtktell( apg->window, "Duplicate invite rejected" ); + } else { + CurGameInfo gi = {0}; + gi_copy( MPPARM(params->mpool) &gi, ¶ms->pgi ); - gi_setNPlayers( &gi, invite->nPlayersT, invite->nPlayersH ); - gi.gameID = invite->gameID; - gi.dictLang = invite->lang; - gi.forceChannel = invite->forceChannel; - replaceStringIfDifferent( params->mpool, &gi.dictName, invite->dict ); + gi_setNPlayers( &gi, invite->nPlayersT, invite->nPlayersH ); + gi.gameID = gameID; + gi.dictLang = invite->lang; + gi.forceChannel = invite->forceChannel; + replaceStringIfDifferent( params->mpool, &gi.dictName, invite->dict ); - GtkGameGlobals* globals = malloc( sizeof(*globals) ); - params->needsNewGame = XP_FALSE; - initGlobals( globals, params, &gi ); + GtkGameGlobals* globals = malloc( sizeof(*globals) ); + params->needsNewGame = XP_FALSE; + initGlobals( globals, params, &gi ); - invit_makeAddrRec( invite, &globals->cGlobals.addr ); - // globals->cGlobals.addr = *returnAddr; + invit_makeAddrRec( invite, &globals->cGlobals.addr ); + // globals->cGlobals.addr = *returnAddr; - GtkWidget* gameWindow = globals->window; - globals->cGlobals.pDb = apg->params->pDb; - globals->cGlobals.selRow = -1; - recordOpened( apg, globals ); - gtk_widget_show( gameWindow ); + GtkWidget* gameWindow = globals->window; + globals->cGlobals.pDb = apg->params->pDb; + globals->cGlobals.selRow = -1; + recordOpened( apg, globals ); + gtk_widget_show( gameWindow ); - gi_disposePlayerInfo( MPPARM(params->mpool) &gi ); + gi_disposePlayerInfo( MPPARM(params->mpool) &gi ); + } } static void @@ -646,17 +738,6 @@ onGameSaved( void* closure, sqlite3_int64 rowid, } } -sqlite3_int64 -getSelRow( const GtkAppGlobals* apg ) -{ - sqlite3_int64 result = -1; - guint len = apg->selRows->len; - if ( 1 == len ) { - result = g_array_index( apg->selRows, sqlite3_int64, 0 ); - } - return result; -} - static GtkAppGlobals* g_globals_for_signal = NULL; static void diff --git a/xwords4/linux/gtkmain.h b/xwords4/linux/gtkmain.h index 547ce5df4..20595be78 100644 --- a/xwords4/linux/gtkmain.h +++ b/xwords4/linux/gtkmain.h @@ -26,5 +26,7 @@ int gtkmain( LaunchParams* params ); void windowDestroyed( GtkGameGlobals* globals ); void onGameSaved( void* closure, sqlite3_int64 rowid, XP_Bool firstTime ); +void open_row( GtkAppGlobals* apg, sqlite3_int64 row, XP_Bool isNew ); +void make_rematch( GtkAppGlobals* apg, const CommonGlobals* cGlobals ); #endif diff --git a/xwords4/linux/linuxmain.c b/xwords4/linux/linuxmain.c index 92d6b57e8..44366033f 100644 --- a/xwords4/linux/linuxmain.c +++ b/xwords4/linux/linuxmain.c @@ -918,6 +918,10 @@ linux_getDevID( LaunchParams* params, DevIDType* typ ) if ( !!params->lDevID ) { result = params->lDevID; *typ = ID_TYPE_LINUX; + } else if ( db_fetch( params->pDb, KEY_RDEVID, params->devIDStore, + sizeof(params->devIDStore) ) ) { + result = params->devIDStore; + *typ = '\0' == result[0] ? ID_TYPE_ANON : ID_TYPE_RELAY; } else if ( db_fetch( params->pDb, KEY_LDEVID, params->devIDStore, sizeof(params->devIDStore) ) ) { result = params->devIDStore; diff --git a/xwords4/linux/main.h b/xwords4/linux/main.h index 636b81aee..838e3cc87 100644 --- a/xwords4/linux/main.h +++ b/xwords4/linux/main.h @@ -247,9 +247,8 @@ typedef struct _GtkAppGlobals { GtkWidget* window; GtkWidget* listWidget; GtkWidget* openButton; + GtkWidget* rematchButton; GtkWidget* deleteButton; } GtkAppGlobals; -sqlite3_int64 getSelRow( const GtkAppGlobals* apg ); - #endif diff --git a/xwords4/linux/relaycon.c b/xwords4/linux/relaycon.c index 34a189020..d5c607753 100644 --- a/xwords4/linux/relaycon.c +++ b/xwords4/linux/relaycon.c @@ -48,14 +48,14 @@ static ssize_t sendIt( RelayConStorage* storage, const XP_U8* msgbuf, XP_U16 len static size_t addVLIStr( XP_U8* buf, size_t len, const XP_UCHAR* str ); static void getNetString( const XP_U8** ptr, XP_U16 len, XP_UCHAR* buf ); static XP_U16 getNetShort( const XP_U8** ptr ); -#ifdef DEBUG static XP_U32 getNetLong( const XP_U8** ptr ); -#endif static int writeHeader( RelayConStorage* storage, XP_U8* dest, XWRelayReg cmd ); static bool readHeader( const XP_U8** buf, MsgHeader* header ); static size_t writeDevID( XP_U8* buf, size_t len, const XP_UCHAR* str ); static size_t writeShort( XP_U8* buf, size_t len, XP_U16 shrt ); static size_t writeLong( XP_U8* buf, size_t len, XP_U32 lng ); +static size_t writeBytes( XP_U8* buf, size_t len, const XP_U8* bytes, + size_t nBytes ); static size_t writeVLI( XP_U8* out, uint32_t nn ); static size_t un2vli( int nn, uint8_t* buf ); static bool vli2un( const uint8_t** inp, uint32_t* outp ); @@ -113,7 +113,8 @@ relaycon_reg( LaunchParams* params, const XP_UCHAR* rDevID, } void -relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invit ) +relaycon_invite( LaunchParams* params, XP_U32 destDevID, + const XP_UCHAR* relayID, InviteInfo* invit ) { XP_U8 tmpbuf[256]; int indx = 0; @@ -122,7 +123,17 @@ relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invit ) indx += writeHeader( storage, tmpbuf, XWPDEV_INVITE ); XP_U32 me = linux_getDevIDRelay( params ); indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, me ); - indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, dest ); + + /* write relayID /, or if we have an actual devID write a + null byte plus it. */ + if ( 0 == destDevID ) { + XP_ASSERT( '\0' != relayID[0] ); + indx += writeBytes( &tmpbuf[indx], sizeof(tmpbuf) - indx, + (XP_U8*)relayID, 1 + XP_STRLEN( relayID ) ); + } else { + tmpbuf[indx++] = '\0'; /* null byte: zero-len str */ + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, destDevID ); + } XWStreamCtxt* stream = mem_stream_make( MPPARM(params->mpool) params->vtMgr, params, @@ -132,8 +143,7 @@ relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invit ) indx += writeShort( &tmpbuf[indx], sizeof(tmpbuf) - indx, len ); XP_ASSERT( indx + len < sizeof(tmpbuf) ); const XP_U8* ptr = stream_getPtr( stream ); - XP_MEMCPY( &tmpbuf[indx], ptr, len ); - indx += len; + indx += writeBytes( &tmpbuf[indx], sizeof(tmpbuf) - indx, ptr, len ); stream_destroy( stream ); sendIt( storage, tmpbuf, indx ); @@ -151,11 +161,8 @@ relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, XP_U8 tmpbuf[1 + 4 + 1 + sizeof(gameToken) + buflen]; int indx = 0; indx += writeHeader( storage, tmpbuf, XWPDEV_MSG ); - XP_U32 inNBO = htonl(gameToken); - XP_MEMCPY( &tmpbuf[indx], &inNBO, sizeof(inNBO) ); - indx += sizeof(inNBO); - XP_MEMCPY( &tmpbuf[indx], buf, buflen ); - indx += buflen; + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, gameToken ); + indx += writeBytes( &tmpbuf[indx], sizeof(tmpbuf) - indx, buf, buflen ); nSent = sendIt( storage, tmpbuf, indx ); if ( nSent > buflen ) { nSent = buflen; @@ -179,14 +186,12 @@ relaycon_sendnoconn( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, 1 + idLen + sizeof(gameToken) + buflen]; indx += writeHeader( storage, tmpbuf, XWPDEV_MSGNOCONN ); - gameToken = htonl( gameToken ); - XP_MEMCPY( &tmpbuf[indx], &gameToken, sizeof(gameToken) ); - indx += sizeof(gameToken); - XP_MEMCPY( &tmpbuf[indx], relayID, idLen ); - indx += idLen; + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, gameToken ); + indx += writeBytes( &tmpbuf[indx], sizeof(tmpbuf) - indx, + (const XP_U8*)relayID, idLen ); tmpbuf[indx++] = '\n'; - XP_MEMCPY( &tmpbuf[indx], buf, buflen ); - nSent = sendIt( storage, tmpbuf, sizeof(tmpbuf) ); + indx += writeBytes( &tmpbuf[indx], sizeof(tmpbuf) - indx, buf, buflen ); + nSent = sendIt( storage, tmpbuf, indx ); if ( nSent > buflen ) { nSent = buflen; } @@ -218,9 +223,7 @@ relaycon_deleted( LaunchParams* params, const XP_UCHAR* devID, int indx = 0; indx += writeHeader( storage, tmpbuf, XWPDEV_DELGAME ); indx += writeDevID( &tmpbuf[indx], sizeof(tmpbuf) - indx, devID ); - gameToken = htonl( gameToken ); - memcpy( &tmpbuf[indx], &gameToken, sizeof(gameToken) ); - indx += sizeof( gameToken ); + indx += writeLong( &tmpbuf[indx], sizeof(tmpbuf) - indx, gameToken ); sendIt( storage, tmpbuf, indx ); } @@ -330,7 +333,10 @@ relaycon_receive( GIOChannel* source, GIOCondition XP_UNUSED_DBG(condition), gpo } case XWPDEV_GOTINVITE: { XP_LOGF( "%s(): got XWPDEV_GOTINVITE", __func__ ); - XP_U32 sender = getNetLong( &ptr ); +#ifdef DEBUG + XP_U32 sender = +#endif + getNetLong( &ptr ); XP_U16 len = getNetShort( &ptr ); XWStreamCtxt* stream = mem_stream_make( MPPARM(storage->params->mpool) storage->params->vtMgr, storage, @@ -438,7 +444,7 @@ writeShort( XP_U8* buf, size_t len, XP_U16 shrt ) { shrt = htons( shrt ); assert( sizeof( shrt ) <= len ); - memcpy( buf, &shrt, sizeof(shrt) ); + XP_MEMCPY( buf, &shrt, sizeof(shrt) ); return sizeof(shrt); } @@ -451,6 +457,14 @@ writeLong( XP_U8* buf, size_t len, XP_U32 lng ) return sizeof(lng); } +static size_t +writeBytes( XP_U8* buf, size_t len, const XP_U8* bytes, size_t nBytes ) +{ + assert( nBytes <= len ); + XP_MEMCPY( buf, bytes, nBytes ); + return nBytes; +} + static size_t writeVLI( XP_U8* out, uint32_t nn ) { @@ -469,7 +483,6 @@ getNetShort( const XP_U8** ptr ) return ntohs( result ); } -#ifdef DEBUG static XP_U32 getNetLong( const XP_U8** ptr ) { @@ -478,7 +491,6 @@ getNetLong( const XP_U8** ptr ) *ptr += sizeof(result); return ntohl( result ); } -#endif static void getNetString( const XP_U8** ptr, XP_U16 len, XP_UCHAR* buf ) diff --git a/xwords4/linux/relaycon.h b/xwords4/linux/relaycon.h index 4a2967659..b835ca230 100644 --- a/xwords4/linux/relaycon.h +++ b/xwords4/linux/relaycon.h @@ -39,7 +39,10 @@ void relaycon_init( LaunchParams* params, const RelayConnProcs* procs, void* procsClosure, const char* host, int port ); void relaycon_reg( LaunchParams* params, const XP_UCHAR* rDevID, DevIDType typ, const XP_UCHAR* devID ); -void relaycon_invite( LaunchParams* params, XP_U32 dest, InviteInfo* invite ); +/* Need one of dest or relayID, with dest preferred. pass 0 for dest to use + relayID (formatted per comms::formatRelayID()) */ +void relaycon_invite( LaunchParams* params, XP_U32 dest, + const XP_UCHAR* relayID, InviteInfo* invite ); XP_S16 relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, XP_U32 gameToken, const CommsAddrRec* addrRec ); XP_S16 relaycon_sendnoconn( LaunchParams* params, const XP_U8* buf, diff --git a/xwords4/relay/dbmgr.cpp b/xwords4/relay/dbmgr.cpp index 6249b48fb..4cc0596b0 100644 --- a/xwords4/relay/dbmgr.cpp +++ b/xwords4/relay/dbmgr.cpp @@ -873,6 +873,19 @@ DBMgr::readArray( const char* const connName, const char* column, int arr[] ) / PQclear( result ); } +// parse something created by comms.c's formatRelayID +DevIDRelay +DBMgr::getDevID( string& relayID ) +{ + size_t pos = relayID.find_first_of( '/' ); + string connName = relayID.substr( 0, pos ); + int hid = relayID[pos + 1] - '0'; + DevIDRelay result = getDevID( connName.c_str(), hid ); + // Not an error. Remove or downlog when confirm working + logf( XW_LOGERROR, "%s(%s) => %d", __func__, relayID.c_str(), result ); + return result; +} + DevIDRelay DBMgr::getDevID( const char* connName, int hid ) { diff --git a/xwords4/relay/dbmgr.h b/xwords4/relay/dbmgr.h index e0fafcaad..84f5977ac 100644 --- a/xwords4/relay/dbmgr.h +++ b/xwords4/relay/dbmgr.h @@ -146,6 +146,8 @@ class DBMgr { void RemoveStoredMessage( const int msgID ); void RemoveStoredMessages( vector& ids ); + DevIDRelay getDevID( string& relayID ); + private: DBMgr(); bool execSql( const string& query ); diff --git a/xwords4/relay/xwrelay.cpp b/xwords4/relay/xwrelay.cpp index 8ea8c03cd..1a13c9b2c 100644 --- a/xwords4/relay/xwrelay.cpp +++ b/xwords4/relay/xwrelay.cpp @@ -1779,17 +1779,24 @@ handle_udp_packet( UdpThreadClosure* utc ) break; } - case XWPDEV_INVITE: + case XWPDEV_INVITE: { DevIDRelay sender; - DevIDRelay invitee; - if ( getNetLong( &ptr, end, &sender ) - && getNetLong( &ptr, end, &invitee) ) { + string relayID; + if ( getNetLong( &ptr, end, &sender ) + && getNetString( &ptr, end, relayID ) ) { + DevIDRelay invitee; + if ( 0 < relayID.size() ) { + invitee = DBMgr::Get()->getDevID( relayID ); + } else if ( !getNetLong( &ptr, end, &invitee ) ) { + break; // failure + } logf( XW_LOGVERBOSE0, "got invite from %d for %d", sender, invitee ); post_invite( sender, invitee, ptr, end - ptr ); } break; - + } + case XWPDEV_KEEPALIVE: case XWPDEV_RQSTMSGS: { DevID devID( ID_TYPE_RELAY ); From 08e6c2b3584b71b7e13e65e5cb4b30a5f3c7dec3 Mon Sep 17 00:00:00 2001 From: Eric House Date: Mon, 6 Jul 2015 20:52:22 -0700 Subject: [PATCH 010/224] use a static to remember last devid typed --- xwords4/linux/gtkinvit.c | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/xwords4/linux/gtkinvit.c b/xwords4/linux/gtkinvit.c index 180473e38..fc8498ab6 100644 --- a/xwords4/linux/gtkinvit.c +++ b/xwords4/linux/gtkinvit.c @@ -32,12 +32,13 @@ typedef struct _PageData { const char* okButtonTxt; } PageData; +static XP_UCHAR s_devIDBuf[32] = {0}; + typedef struct _GtkInviteState { GtkGameGlobals* globals; CommsAddrRec* addr; gint* nPlayersP; gint maxPlayers; - XP_UCHAR devIDBuf[32]; GtkWidget* nPlayersCombo; /* relay */ @@ -93,6 +94,7 @@ handle_ok( GtkWidget* XP_UNUSED(widget), gpointer closure ) #ifdef XWFEATURE_RELAY case COMMS_CONN_RELAY: txt = gtk_entry_get_text( GTK_ENTRY(state->devID) ); + snprintf( s_devIDBuf, sizeof(s_devIDBuf), "%s", txt ); state->addr->u.ip_relay.devID = atoi( txt ); break; #endif @@ -119,7 +121,6 @@ handle_ok( GtkWidget* XP_UNUSED(widget), gpointer closure ) gchar* num = gtk_combo_box_get_active_text( GTK_COMBO_BOX(state->nPlayersCombo) ); *(state->nPlayersP) = atoi( num ); - XP_LOGF( "num players: %d", *(state->nPlayersP) ); state->cancelled = XP_FALSE; gtk_main_quit(); @@ -170,10 +171,7 @@ makeRelayPage( GtkInviteState* state, PageData* data ) GtkWidget* hbox; hbox = makeLabeledField( "Invitee DeviceID", &state->devID, NULL ); - XP_Bool hasRelay = addr_hasType( state->addr, COMMS_CONN_RELAY ); - if ( hasRelay ) { - gtk_entry_set_text( GTK_ENTRY(state->devID), state->devIDBuf ); - } + gtk_entry_set_text( GTK_ENTRY(state->devID), s_devIDBuf ); gtk_box_pack_start( GTK_BOX(vbox), hbox, FALSE, TRUE, 0 ); gtk_widget_show( vbox ); From 1f3959fc71b35bfe50064702794212abb1a0fdf7 Mon Sep 17 00:00:00 2001 From: Eric House Date: Mon, 6 Jul 2015 21:18:51 -0700 Subject: [PATCH 011/224] fix rematch initiated by a guest: needs to be host in the game it creates. --- xwords4/linux/gtkmain.c | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/xwords4/linux/gtkmain.c b/xwords4/linux/gtkmain.c index d4d5e8a8e..2f726a50e 100644 --- a/xwords4/linux/gtkmain.c +++ b/xwords4/linux/gtkmain.c @@ -292,9 +292,14 @@ make_rematch( GtkAppGlobals* apg, const CommonGlobals* cGlobals ) /* Create new game. But has no addressing info, so need to set that aside for later. */ + const CommsCtxt* comms = cGlobals->game.comms; CurGameInfo gi = {0}; gi_copy( MPPARM(cGlobals->util->mpool) &gi, cGlobals->gi ); gi.gameID = 0; /* clear so will get generated */ + if ( !!comms ) { + gi.serverRole = SERVER_ISSERVER; + gi.forceChannel = 0; + } game_saveNewGame( MPPARM(cGlobals->util->mpool) &gi, cGlobals->util, &cGlobals->cp, stream ); @@ -304,7 +309,6 @@ make_rematch( GtkAppGlobals* apg, const CommonGlobals* cGlobals ) /* If it's a multi-device game, save enough information with it than when opened it can invite the other device[s] join the rematch. */ - const CommsCtxt* comms = cGlobals->game.comms; if ( !!comms ) { XWStreamCtxt* stream = mem_stream_make( MPPARM(cGlobals->util->mpool) cGlobals->params->vtMgr, From b5ae54f313c7e306a2ba0a74e280c96bfd34644c Mon Sep 17 00:00:00 2001 From: Eric House Date: Fri, 10 Jul 2015 23:19:38 -0700 Subject: [PATCH 012/224] use common/native NetLaunchInfo to implement invite-via-relay that works between linux and Android clients. Required renaming so struct names and names of fields within match in c and java code. The point is to test this as the foundation of rematch: now you have to type in a deviceID in order to invite, which clearly sucks for users. Either that goes away, or it's replaced with something that scans existing games and lists past opponents as possible invitees. --- xwords4/android/XWords4/AndroidManifest.xml | 5 + xwords4/android/XWords4/archive/R.java | 1568 +++++++++-------- xwords4/android/XWords4/jni/Android.mk | 3 +- xwords4/android/XWords4/jni/andutils.c | 28 + xwords4/android/XWords4/jni/andutils.h | 4 + xwords4/android/XWords4/jni/xwjni.c | 98 +- .../XWords4/res/layout/relayinviter.xml | 58 + .../XWords4/res/values/common_rsrc.xml | 1 + .../android/XWords4/res/values/strings.xml | 19 +- .../XWords4/res_src/values-ba_CK/strings.xml | 20 +- .../XWords4/res_src/values-ca_PS/strings.xml | 20 +- .../XWords4/res_src/values-fr/strings.xml | 2 +- .../XWords4/res_src/values-pt/strings.xml | 2 +- .../XWords4/res_src/values-sk/strings.xml | 2 +- .../eehouse/android/xw4/BoardDelegate.java | 31 +- .../org/eehouse/android/xw4/DlgDelegate.java | 6 +- .../android/xw4/GameConfigDelegate.java | 4 +- .../org/eehouse/android/xw4/GameUtils.java | 3 +- .../eehouse/android/xw4/NetLaunchInfo.java | 58 +- .../android/xw4/RelayInviteActivity.java | 36 + .../android/xw4/RelayInviteDelegate.java | 460 +++++ .../org/eehouse/android/xw4/RelayService.java | 137 +- .../android/xw4/SMSInviteDelegate.java | 2 +- .../src/org/eehouse/android/xw4/XWPrefs.java | 21 + .../eehouse/android/xw4/jni/CommsAddrRec.java | 6 + .../org/eehouse/android/xw4/jni/XwJNI.java | 17 +- xwords4/common/game.c | 7 +- xwords4/common/invit.c | 33 +- xwords4/common/invit.h | 12 +- xwords4/linux/Makefile | 1 + xwords4/linux/gtkboard.c | 4 +- xwords4/linux/gtkmain.c | 2 +- xwords4/linux/relaycon.c | 4 +- xwords4/linux/relaycon.h | 4 +- 34 files changed, 1805 insertions(+), 873 deletions(-) create mode 100644 xwords4/android/XWords4/res/layout/relayinviter.xml create mode 100644 xwords4/android/XWords4/src/org/eehouse/android/xw4/RelayInviteActivity.java create mode 100644 xwords4/android/XWords4/src/org/eehouse/android/xw4/RelayInviteDelegate.java diff --git a/xwords4/android/XWords4/AndroidManifest.xml b/xwords4/android/XWords4/AndroidManifest.xml index 20c206b94..59f7b391b 100644 --- a/xwords4/android/XWords4/AndroidManifest.xml +++ b/xwords4/android/XWords4/AndroidManifest.xml @@ -102,6 +102,11 @@ android:configChanges="keyboardHidden|orientation|screenSize" android:screenOrientation="sensor" /> + Turn Bluetooth on In the Bluetooth invite device dialog */ - public static final int bt_pick_addall_button=0x7f0501e1; + public static final int bt_pick_addall_button=0x7f0501e3; /** */ - public static final int bt_pick_clear_button=0x7f0501e2; + public static final int bt_pick_clear_button=0x7f0501e4; /** */ - public static final int bt_resend_fmt=0x7f0501ef; + public static final int bt_resend_fmt=0x7f0501f1; /** text of button for adding new player to game */ - public static final int button_add_player=0x7f0500b1; + public static final int button_add_player=0x7f0500b2; /** The only button available when the above message is displayed */ - public static final int button_close_game=0x7f05018b; - public static final int button_decline=0x7f05023c; + public static final int button_close_game=0x7f05018d; + public static final int button_decline=0x7f050243; /** */ - public static final int button_default_both=0x7f0500a7; + public static final int button_default_both=0x7f0500a8; /** These three strings are the text for three buttons giving choices in respose to the dialog launched in response to the dicts_item_select menu (see dicts_item_select above.) */ - public static final int button_default_human=0x7f0500a5; + public static final int button_default_human=0x7f0500a6; /** */ - public static final int button_default_robot=0x7f0500a6; + public static final int button_default_robot=0x7f0500a7; /** When you select the gamel_menu_delete_all menuitem or list_item_delete contextual menu, you are asked to confirm. This is the text of the first button ("Cancel" is the second). */ - public static final int button_delete=0x7f050089; + public static final int button_delete=0x7f05008a; /** Used for button in dialog put up with the relay says the remote partner of this game has deleted it. Ok, meaning "don't discard", is the other option. This same button is @@ -641,41 +642,41 @@ public final class R { the other option, for discarding changes that would otherwise reset a game. */ - public static final int button_discard=0x7f050198; - public static final int button_discard_changes=0x7f0502c8; + public static final int button_discard=0x7f05019a; + public static final int button_discard_changes=0x7f0502cf; /** */ - public static final int button_done=0x7f0501c8; + public static final int button_done=0x7f0501ca; /** */ - public static final int button_done_fmt=0x7f0501c9; + public static final int button_done_fmt=0x7f0501cb; /** Text of button displayed when downloading is an option */ - public static final int button_download=0x7f05018e; - public static final int button_edit=0x7f0502c7; - public static final int button_enable=0x7f050307; - public static final int button_enable_bt=0x7f050227; - public static final int button_enable_sms=0x7f050226; - public static final int button_go_settings=0x7f05025e; + public static final int button_download=0x7f050190; + public static final int button_edit=0x7f0502ce; + public static final int button_enable=0x7f05030e; + public static final int button_enable_bt=0x7f05022e; + public static final int button_enable_sms=0x7f05022d; + public static final int button_go_settings=0x7f050265; /** */ - public static final int button_invite=0x7f0501f1; + public static final int button_invite=0x7f0501f3; /** text of button to juggle (randomly rearrange order of) players */ - public static final int button_juggle_players=0x7f0500b2; - public static final int button_later=0x7f050228; + public static final int button_juggle_players=0x7f0500b3; + public static final int button_later=0x7f05022f; /** New strings that need to be documented and found a home above. */ - public static final int button_lookup=0x7f0501c2; + public static final int button_lookup=0x7f0501c4; /** */ - public static final int button_lookup_fmt=0x7f0501c4; - public static final int button_lookup_study=0x7f0501c3; - public static final int button_lookup_study_fmt=0x7f0501c5; + public static final int button_lookup_fmt=0x7f0501c6; + public static final int button_lookup_study=0x7f0501c5; + public static final int button_lookup_study_fmt=0x7f0501c7; /** */ - public static final int button_move=0x7f0501cd; + public static final int button_move=0x7f0501cf; /** What is \u200C? English strings are used as keys, so they all need to be unique. This glyph is non-printing and of zero-width, so it should do the trick: @@ -716,32 +717,35 @@ public final class R { Any changes you make to it will be lost. */ - public static final int button_new_game=0x7f05007f; + public static final int button_new_game=0x7f050080; /** Games list screen menuitem text for creating a new group */ - public static final int button_new_group=0x7f050080; - public static final int button_newgroup=0x7f0501ce; - public static final int button_no=0x7f050196; + public static final int button_new_group=0x7f050081; + public static final int button_newgroup=0x7f0501d0; + public static final int button_no=0x7f050198; /** Text for button in new-user-info dialog with title just above. */ - public static final int button_notagain=0x7f0501a4; - public static final int button_reconnect=0x7f05024e; + public static final int button_notagain=0x7f0501a6; + public static final int button_reconnect=0x7f050255; + /** + */ + public static final int button_relay_add=0x7f050200; /** Button shown in game over dialog triggering creation of new game with the same players and parameters as the one that just ended. */ - public static final int button_rematch=0x7f05024d; + public static final int button_rematch=0x7f050254; /** When you select the list_item_reset contextual menu, you are asked to confirm. This is the text of the first button ("Cancel" is the second). */ - public static final int button_reset=0x7f05008a; + public static final int button_reset=0x7f05008b; /** Used for a button when informing user that his attempt to connect to the relay failed because the room named does not exist. (I believe this no longer occurs.) */ - public static final int button_retry=0x7f050199; + public static final int button_retry=0x7f05019b; /** ############################################################ # :Menus: @@ -754,25 +758,25 @@ public final class R { text of first menu item. Will revert all preferences to their default/original values */ - public static final int button_revert_all=0x7f05010f; + public static final int button_revert_all=0x7f050110; /** Second menu item. Reverts only the colors to their default/original values */ - public static final int button_revert_colors=0x7f050111; + public static final int button_revert_colors=0x7f050112; /** Used in Game config dialog to confirm saving changes that reset a game */ - public static final int button_save=0x7f050197; + public static final int button_save=0x7f050199; /** */ - public static final int button_search=0x7f0501cf; + public static final int button_search=0x7f0501d1; /** */ - public static final int button_sms_add=0x7f0501fc; + public static final int button_sms_add=0x7f0501ff; /** Text of button allowing user to choose to open with a different (but same-language wordlist) */ - public static final int button_substdict=0x7f05018f; - public static final int button_trade_cancel=0x7f0500d9; + public static final int button_substdict=0x7f050191; + public static final int button_trade_cancel=0x7f0500da; /** ############################################################ # :Screens: @@ -786,7 +790,7 @@ public final class R { Buttons shown at bottom when board is in exchange mode, i.e. after user has picked board_menu_trade menu item */ - public static final int button_trade_commit=0x7f0500d8; + public static final int button_trade_commit=0x7f0500d9; /** ############################################################ # Dialogs @@ -795,20 +799,20 @@ public final class R { Text for buttons at the bottom of dialogs. These first are in many places. */ - public static final int button_yes=0x7f050195; - public static final int cannot_delete_default_group_fmt=0x7f05024c; - public static final int change_group=0x7f050251; + public static final int button_yes=0x7f050197; + public static final int cannot_delete_default_group_fmt=0x7f050253; + public static final int change_group=0x7f050258; /** text of button in About Crosswords dialog summoning above dialog */ - public static final int changes_button=0x7f0501c1; + public static final int changes_button=0x7f0501c3; /** text of dialog showing the set of changes made since the last release */ - public static final int changes_title=0x7f0501c0; + public static final int changes_title=0x7f0501c2; /** Prefix for local messages */ - public static final int chat_local_id=0x7f050175; + public static final int chat_local_id=0x7f050177; /** ############################################################ # :Menus: @@ -820,14 +824,14 @@ public final class R { other devices in the game: no "clear chat" message is transmitted. */ - public static final int chat_menu_clear=0x7f050178; + public static final int chat_menu_clear=0x7f05017a; /** Prefix for remote messages */ - public static final int chat_other_id=0x7f050176; + public static final int chat_other_id=0x7f050178; /** Text on the button that causes the contents of the message-composition field to be sent. */ - public static final int chat_send=0x7f050177; + public static final int chat_send=0x7f050179; /** ############################################################ # :Screens: # Chat screen @@ -842,70 +846,71 @@ public final class R { title of the chat screen. The name of the current game is substituted for %1$s. */ - public static final int chat_title_fmt=0x7f050174; - public static final int checking_for_fmt=0x7f0502fe; - public static final int checking_title=0x7f0502fd; - public static final int checkupdates_none_found=0x7f05022a; + public static final int chat_title_fmt=0x7f050176; + public static final int checking_for_fmt=0x7f050305; + public static final int checking_title=0x7f050304; + public static final int checkupdates_none_found=0x7f050231; /** color of the "crosshairs", lines drawn vertically and horizontally through the square the user is currently touching in order to guide the fat-fingered (most of us) in operations that require accurately selecting a single square on the board. */ - public static final int clr_crosshairs=0x7f05013d; - public static final int config_no_connvia=0x7f0502c6; + public static final int clr_crosshairs=0x7f05013e; + public static final int config_no_connvia=0x7f0502cd; + public static final int confirm_clear_relay=0x7f050208; /** */ - public static final int confirm_clear=0x7f050201; + public static final int confirm_clear_sms=0x7f050207; /** Additional text appended to text confirm_delete_dictf in the wordlist delete confiration dialog in the case where the wordlist to be deleted is the last in its language. The name of the language is substituted for %1$s. */ - public static final int confirm_deleteonly_dicts_fmt=0x7f0500a3; - public static final int confirm_drop_relay=0x7f050303; - public static final int confirm_drop_relay_bt=0x7f050305; - public static final int confirm_drop_relay_sms=0x7f050306; - public static final int confirm_get_locdict_fmt=0x7f0502cd; + public static final int confirm_deleteonly_dicts_fmt=0x7f0500a4; + public static final int confirm_drop_relay=0x7f05030a; + public static final int confirm_drop_relay_bt=0x7f05030c; + public static final int confirm_drop_relay_sms=0x7f05030d; + public static final int confirm_get_locdict_fmt=0x7f0502d4; /** Text of confirmation dialog for above */ - public static final int confirm_revert_all=0x7f050112; + public static final int confirm_revert_all=0x7f050113; /** Text of confirmation dialog for above */ - public static final int confirm_revert_colors=0x7f050110; + public static final int confirm_revert_colors=0x7f050111; /** title of confirmation dialog put up when user has unlocked an in-progress game and wants to save changes (has hit the play button). */ - public static final int confirm_save=0x7f0500cb; + public static final int confirm_save=0x7f0500cc; /** title of above confirmation dialog */ - public static final int confirm_save_title=0x7f0500cc; + public static final int confirm_save_title=0x7f0500cd; /** */ - public static final int confirm_sms_expl=0x7f05021f; + public static final int confirm_sms_expl=0x7f050226; /** */ - public static final int confirm_sms_leave=0x7f050221; + public static final int confirm_sms_leave=0x7f050228; /** */ - public static final int confirm_sms_prompt=0x7f050220; + public static final int confirm_sms_prompt=0x7f050227; /** */ - public static final int confirm_sms_title=0x7f05021e; + public static final int confirm_sms_title=0x7f050225; /** */ - public static final int confirm_sms_unlimited=0x7f050222; + public static final int confirm_sms_unlimited=0x7f050229; /** */ - public static final int confirm_sms_willpay=0x7f050223; + public static final int confirm_sms_willpay=0x7f05022a; /** text of dialog shown when the menu item board_menu_undo_last is chosen. */ - public static final int confirm_undo_last=0x7f05010d; - public static final int connect_daily=0x7f05015c; - public static final int connect_fifteen_mins=0x7f050158; - public static final int connect_five_mins=0x7f050157; + public static final int confirm_undo_last=0x7f05010e; + public static final int connect_daily=0x7f05015d; + public static final int connect_fifteen_mins=0x7f050159; + public static final int connect_five_mins=0x7f050158; /** ############################################################ # :Screens: @@ -918,81 +923,81 @@ public final class R { new moves available and fetches them. This controls how frequently that check is done. */ - public static final int connect_frequency=0x7f050155; + public static final int connect_frequency=0x7f050156; /** text of separator marking out the connection area of the dialog */ - public static final int connect_label_fmt=0x7f0500b5; + public static final int connect_label_fmt=0x7f0500b6; /** */ - public static final int connect_label_sms=0x7f050202; + public static final int connect_label_sms=0x7f050209; /** These are the possible values for the connect_frequency setting presented as a drop-down list. */ - public static final int connect_never=0x7f050156; - public static final int connect_one_hour=0x7f05015a; - public static final int connect_six_hours=0x7f05015b; - public static final int connect_thirty_mins=0x7f050159; - public static final int connection_via_label=0x7f0502bf; + public static final int connect_never=0x7f050157; + public static final int connect_one_hour=0x7f05015b; + public static final int connect_six_hours=0x7f05015c; + public static final int connect_thirty_mins=0x7f05015a; + public static final int connection_via_label=0x7f0502c6; /** */ - public static final int connstat_lastother_succ_fmt=0x7f050216; + public static final int connstat_lastother_succ_fmt=0x7f05021d; /** */ - public static final int connstat_lastother_unsucc_fmt=0x7f050217; + public static final int connstat_lastother_unsucc_fmt=0x7f05021e; /** */ - public static final int connstat_lastreceipt_fmt=0x7f050218; + public static final int connstat_lastreceipt_fmt=0x7f05021f; /** */ - public static final int connstat_lastsend_fmt=0x7f050215; + public static final int connstat_lastsend_fmt=0x7f05021c; /** */ - public static final int connstat_net_fmt=0x7f050212; - public static final int connstat_net_noaddr=0x7f050211; + public static final int connstat_net_fmt=0x7f050219; + public static final int connstat_net_noaddr=0x7f050218; /** */ - public static final int connstat_noreceipt=0x7f050219; + public static final int connstat_noreceipt=0x7f050220; /** */ - public static final int connstat_relay=0x7f05021a; + public static final int connstat_relay=0x7f050221; /** */ - public static final int connstat_sms=0x7f05021b; + public static final int connstat_sms=0x7f050222; /** */ - public static final int connstat_succ=0x7f050213; + public static final int connstat_succ=0x7f05021a; /** */ - public static final int connstat_unsucc=0x7f050214; + public static final int connstat_unsucc=0x7f05021b; /** title of dialog brought up in response to the board_menu_game_counts menu. The dialog lists all tiles in the language being used for the game together with how many of each there are are and how many points each is worth. */ - public static final int counts_values_title=0x7f05019e; - public static final int cur_menu_marker_fmt=0x7f05025b; + public static final int counts_values_title=0x7f0501a0; + public static final int cur_menu_marker_fmt=0x7f050262; /** */ - public static final int cur_tiles_fmt=0x7f0501d3; - public static final int data_gsm_only=0x7f050278; - public static final int db_store_done=0x7f0502ff; - public static final int debug_features=0x7f0502d5; - public static final int debug_features_summary=0x7f0502d6; + public static final int cur_tiles_fmt=0x7f0501d5; + public static final int data_gsm_only=0x7f05027f; + public static final int db_store_done=0x7f050306; + public static final int debug_features=0x7f0502dc; + public static final int debug_features_summary=0x7f0502dd; /** dictionary used by default for human players when creating new game */ - public static final int default_dict=0x7f05011f; + public static final int default_dict=0x7f050120; /** other */ - public static final int default_host=0x7f050076; + public static final int default_host=0x7f050077; /** label within default wordlists in app preferences */ - public static final int default_language=0x7f0502af; - public static final int default_loc=0x7f05023f; - public static final int default_loc_summary=0x7f050240; + public static final int default_language=0x7f0502b6; + public static final int default_loc=0x7f050246; + public static final int default_loc_summary=0x7f050247; /** Welcome dialog text */ - public static final int default_name_message=0x7f0501ba; + public static final int default_name_message=0x7f0501bc; /** ########################################################### # :Dialogs: @@ -1007,39 +1012,39 @@ public final class R { Welcome dialog title */ - public static final int default_name_title=0x7f0501b9; + public static final int default_name_title=0x7f0501bb; /** default new-game setting for handline phonies (words not found in the word list) */ - public static final int default_phonies=0x7f050125; + public static final int default_phonies=0x7f050126; /** dictionary used by default for robot players when creating new game */ - public static final int default_robodict=0x7f050120; - public static final int default_update_url=0x7f05007c; - public static final int delete_dicts=0x7f05029a; + public static final int default_robodict=0x7f050121; + public static final int default_update_url=0x7f05007d; + public static final int delete_dicts=0x7f0502a1; /** Playing via SMS is currently disabled. You can enable it in Settings->Network game settings. */ - public static final int dft_sms_name_fmt=0x7f0501f9; + public static final int dft_sms_name_fmt=0x7f0501fc; /** */ - public static final int dict_browse_nowords_fmt=0x7f0501d5; - public static final int dict_desc_fmt=0x7f0502a1; - public static final int dict_host=0x7f0502d2; - public static final int dict_info_fmt=0x7f05029e; + public static final int dict_browse_nowords_fmt=0x7f0501d7; + public static final int dict_desc_fmt=0x7f0502a8; + public static final int dict_host=0x7f0502d9; + public static final int dict_info_fmt=0x7f0502a5; /** label for dropdown by which wordlist is chosen that this player will use. The language the game will use (which constrains the choice of wordlists) is substituted in for "%1$s". */ - public static final int dict_lang_label_fmt=0x7f0500d4; - public static final int dict_on_server=0x7f0502a0; + public static final int dict_lang_label_fmt=0x7f0500d5; + public static final int dict_on_server=0x7f0502a7; /** string name="invite_mime">text/plainDisplay snapshots of games */ - public static final int dropped_dupe=0x7f05025a; + public static final int dropped_dupe=0x7f050261; /** Shown in the main screen when you launch Crosswords from an invitation (received in email or messaging app, say) and there's already a game running that matches that invitation. @@ -1088,64 +1093,67 @@ public final class R { the same room name over and over so they'll get this warning and it's harmless to ignore it. */ - public static final int dup_game_query_fmt=0x7f05019a; + public static final int dup_game_query_fmt=0x7f05019c; /** */ - public static final int email_author_chooser=0x7f0501df; + public static final int email_author_chooser=0x7f0501e1; /** Nor is my email address */ - public static final int email_author_email=0x7f050075; + public static final int email_author_email=0x7f050076; /** */ - public static final int email_author_subject=0x7f0501de; + public static final int email_author_subject=0x7f0501e0; /** */ - public static final int email_body_rev_fmt=0x7f0501e0; + public static final int email_body_rev_fmt=0x7f0501e2; /** color of empty squares on the board (that are not bonus squares) */ - public static final int empty=0x7f05013f; + public static final int empty=0x7f050140; + /** + */ + public static final int empty_relay_inviter=0x7f050204; /** Shows in SMS Invite dialog when no phone numbers have been saved previously */ - public static final int empty_sms_inviter=0x7f0501ff; - public static final int enable_dupes_summary=0x7f0502e3; - public static final int enable_dupes_title=0x7f0502e1; - public static final int enable_nfc=0x7f05025d; - public static final int enable_nfc_toself_summary=0x7f0502e6; - public static final int enable_nfc_toself_title=0x7f0502e5; - public static final int enable_pubroom_summary=0x7f0502be; - public static final int enable_pubroom_title=0x7f0502bd; + public static final int empty_sms_inviter=0x7f050203; + public static final int enable_dupes_summary=0x7f0502ea; + public static final int enable_dupes_title=0x7f0502e8; + public static final int enable_nfc=0x7f050264; + public static final int enable_nfc_toself_summary=0x7f0502ed; + public static final int enable_nfc_toself_title=0x7f0502ec; + public static final int enable_pubroom_summary=0x7f0502c5; + public static final int enable_pubroom_title=0x7f0502c4; /** */ - public static final int enable_sms=0x7f05021c; + public static final int enable_sms=0x7f050223; /** */ - public static final int enable_sms_summary=0x7f05021d; - public static final int enable_sms_toself_summary=0x7f0502e8; - public static final int enable_sms_toself_title=0x7f0502e7; - public static final int entering_trade=0x7f0500da; - public static final int err_dup_invite_fmt=0x7f0502b8; - public static final int expl_update_url=0x7f0502f2; + public static final int enable_sms_summary=0x7f050224; + public static final int enable_sms_toself_summary=0x7f0502ef; + public static final int enable_sms_toself_title=0x7f0502ee; + public static final int entering_trade=0x7f0500db; + public static final int err_dup_invite_fmt=0x7f0502bf; + public static final int expl_update_url=0x7f0502f9; /** if this preference is checked, a dialog will be posted every time a robot makes a move or a move is received from a remote player. */ - public static final int explain_robot=0x7f050147; + public static final int explain_robot=0x7f050148; /** explanation of the above */ - public static final int explain_robot_summary=0x7f050148; + public static final int explain_robot_summary=0x7f050149; /** title of dialog brought up in response to the board_menu_game_final menu. The dialog displays the final score and an accounting of it (including subtractions for running out the game timer if there is one.) */ - public static final int finalscores_title=0x7f0501a0; + public static final int finalscores_title=0x7f0501a2; /** Explanatory text appears in the dialog */ - public static final int force_expl=0x7f0500cf; - public static final int force_radio_title=0x7f0502e9; - public static final int force_tablet_summary=0x7f0502a5; - public static final int force_tablet_title=0x7f0502a4; + public static final int force_expl=0x7f0500d0; + public static final int force_radio_title=0x7f0502f0; + public static final int force_tablet_summary=0x7f0502ac; + public static final int force_tablet_title=0x7f0502ab; /** ############################################################ # :Dialogs: @@ -1160,22 +1168,22 @@ public final class R { title of this dialog */ - public static final int force_title=0x7f0500ce; + public static final int force_title=0x7f0500cf; /** Displayed as a "Toast" (mini window briefly displayed) if you didn't choose a remote player and I was forced to pick one. I always pick the first. */ - public static final int forced_consistent=0x7f0500d0; + public static final int forced_consistent=0x7f0500d1; /** Games that have ended are listed with this string */ - public static final int gameOver=0x7f050088; + public static final int gameOver=0x7f050089; /** */ - public static final int game_btname_title=0x7f0501f4; + public static final int game_btname_title=0x7f0501f7; /** used to create default names of games (when user has not named them.) */ - public static final int game_fmt=0x7f050192; + public static final int game_fmt=0x7f050194; /** ############################################################ # :Menus: @@ -1185,49 +1193,49 @@ public final class R { title of contextual menu. Name of the selected game is substituted in. */ - public static final int game_item_menu_title_fmt=0x7f050091; + public static final int game_item_menu_title_fmt=0x7f050092; /** */ - public static final int game_list_tmp=0x7f050210; + public static final int game_list_tmp=0x7f050217; /** text of checkbox at top of dialog allowing to unlock in-play game to make changes */ - public static final int game_locked=0x7f0500ae; - public static final int game_name_group_title=0x7f05024b; + public static final int game_locked=0x7f0500af; + public static final int game_name_group_title=0x7f050252; /** */ - public static final int game_name_label=0x7f0501f6; + public static final int game_name_label=0x7f0501f9; /** */ - public static final int game_name_title=0x7f0501f5; + public static final int game_name_title=0x7f0501f8; /** */ - public static final int game_rename_title=0x7f05009a; + public static final int game_rename_title=0x7f05009b; /** Put nothing in the summary space, so it just reads "Game 2" */ - public static final int game_summary_field_empty=0x7f05012b; - public static final int game_summary_field_gameid=0x7f0502f0; + public static final int game_summary_field_empty=0x7f05012c; + public static final int game_summary_field_gameid=0x7f0502f7; /** Put the language there, so it reads "Game 2 (English)" */ - public static final int game_summary_field_language=0x7f05012c; - public static final int game_summary_field_npackets=0x7f0502f1; + public static final int game_summary_field_language=0x7f05012d; + public static final int game_summary_field_npackets=0x7f0502f8; /** List names of opponents (summarized), e.g. "Game 2 (vs Kati)" */ - public static final int game_summary_field_opponents=0x7f05012d; - public static final int game_summary_field_rowid=0x7f0502ef; + public static final int game_summary_field_opponents=0x7f05012e; + public static final int game_summary_field_rowid=0x7f0502f6; /** List the state of the game, "Game over" or "10 moves made" etc. */ - public static final int game_summary_field_state=0x7f05012e; + public static final int game_summary_field_state=0x7f05012f; /** Regardless of the setting of the connect_frequency preference, checks the relay immediately for any moves for networked games on this device and posts a notification if any is downloaded. */ - public static final int gamel_menu_checkmoves=0x7f05008d; + public static final int gamel_menu_checkmoves=0x7f05008e; /** */ - public static final int gamel_menu_checkupdates=0x7f050229; + public static final int gamel_menu_checkupdates=0x7f050230; /** ############################################################ # :Menus: @@ -1239,25 +1247,26 @@ public final class R { Brings up the Wordlists (formerly Dictionaries) screen */ - public static final int gamel_menu_dicts=0x7f05008b; - public static final int gamel_menu_loaddb=0x7f0502e0; - public static final int gamel_menu_storedb=0x7f0502df; - public static final int gamel_menu_study=0x7f050266; + public static final int gamel_menu_dicts=0x7f05008c; + public static final int gamel_menu_loaddb=0x7f0502e7; + public static final int gamel_menu_storedb=0x7f0502e6; + public static final int gamel_menu_study=0x7f05026d; + public static final int get_relay_number=0x7f050206; /** */ - public static final int get_sms_number=0x7f050200; - public static final int getinfo=0x7f050299; - public static final int git_rev=0x7f05007e; - public static final int git_rev_title=0x7f0502db; - public static final int got_langdict_summary=0x7f0502f4; - public static final int got_langdict_title=0x7f0502f3; - public static final int green=0x7f050143; - public static final int group_cur_games=0x7f050248; - public static final int group_new_games=0x7f050249; + public static final int get_sms_number=0x7f050205; + public static final int getinfo=0x7f0502a0; + public static final int git_rev=0x7f05007f; + public static final int git_rev_title=0x7f0502e2; + public static final int got_langdict_summary=0x7f0502fb; + public static final int got_langdict_title=0x7f0502fa; + public static final int green=0x7f050144; + public static final int group_cur_games=0x7f05024f; + public static final int group_new_games=0x7f050250; /** Used as the default name for remote players displayed within the Game configure screen */ - public static final int guest_name=0x7f0500ca; + public static final int guest_name=0x7f0500cb; /** If this preference is checked the "crosshairs" (vertical and horzontal lines through the cell your finger is on that help you tell where the app thinks you're actually tapping) will @@ -1266,75 +1275,76 @@ public final class R { large enough screens that they always know where they're tapping. */ - public static final int hide_crosshairs=0x7f050151; + public static final int hide_crosshairs=0x7f050152; /** explanation of the above */ - public static final int hide_crosshairs_summary=0x7f050152; + public static final int hide_crosshairs_summary=0x7f050153; /** clarification of above */ - public static final int hide_newgames_summary=0x7f050132; + public static final int hide_newgames_summary=0x7f050133; /** Checkbox that when set prevents showing the newgame buttons on the main screen to save space */ - public static final int hide_newgames_title=0x7f050131; + public static final int hide_newgames_title=0x7f050132; /** Checkbox that when set prevents showing a title bar in the game board window to save space */ - public static final int hide_title=0x7f05012f; + public static final int hide_title=0x7f050130; /** clarification of above */ - public static final int hide_title_summary=0x7f050130; + public static final int hide_title_summary=0x7f050131; /** text of checkbox. If this checkbox is checked, buttons will appear by which the user can get high- and low-scoring moves generated based on his tray. Can be used to cheat. */ - public static final int hints_allowed=0x7f0500bd; + public static final int hints_allowed=0x7f0500be; /** clarification on hints_allowed, whether new games will default to having the hint feature enabled(string elsewhere in this file) */ - public static final int hints_allowed_sum=0x7f050121; + public static final int hints_allowed_sum=0x7f050122; /** title of dialog brought up in response to the board_menu_game_history menu. A full history of the game up to the last turn is displayed, though details about what's in players' racks is left out if the game is not yet over. */ - public static final int history_title=0x7f05019f; + public static final int history_title=0x7f0501a1; /** Partial text of alert posted when phonies_warn or phonies_disallow is the current setting and a "phony" is played. One of the two following strings will be appended */ - public static final int ids_badwords_fmt=0x7f0500ea; + public static final int ids_badwords_fmt=0x7f0500eb; /** text of confirmation dialog shown when user chooses the menu item with text board_menu_game_final and the game is not over */ - public static final int ids_endnow=0x7f0501a1; + public static final int ids_endnow=0x7f0501a3; /** Title of generic dialog used to display information */ - public static final int info_title=0x7f05019b; + public static final int info_title=0x7f05019d; /** */ - public static final int inform_dict_diffdict_fmt=0x7f050234; + public static final int inform_dict_diffdict_fmt=0x7f05023b; /** */ - public static final int inform_dict_diffversion_fmt=0x7f050232; - public static final int inform_dict_download=0x7f050235; - public static final int inform_dict_title=0x7f050233; + public static final int inform_dict_diffversion_fmt=0x7f050239; + public static final int inform_dict_download=0x7f05023c; + public static final int inform_dict_title=0x7f05023a; /** Will new games, on default, randomly rearrange the start order of players. */ - public static final int init_autojuggle=0x7f050123; + public static final int init_autojuggle=0x7f050124; /** clarification on above */ - public static final int init_autojuggle_sum=0x7f050124; + public static final int init_autojuggle_sum=0x7f050125; /** default number of minutes on timer for new games */ - public static final int initial_player_minutes=0x7f050126; - public static final int invite_choice_bt=0x7f05016b; - public static final int invite_choice_email=0x7f05016a; - public static final int invite_choice_nfc=0x7f05016c; + public static final int initial_player_minutes=0x7f050127; + public static final int invite_choice_bt=0x7f05016c; + public static final int invite_choice_email=0x7f05016b; + public static final int invite_choice_nfc=0x7f05016d; + public static final int invite_choice_relay=0x7f05016e; /** EXPLAIN ME Bluetooth is not available. This may mean that your device doesn\'t support it, or that it\'s been @@ -1343,9 +1353,9 @@ public final class R { between html and plaintext formatting but I also provide some explanation/guidance. */ - public static final int invite_choice_sms=0x7f050169; - public static final int invite_choice_title=0x7f05016d; - public static final int invite_chooser_email=0x7f050172; + public static final int invite_choice_sms=0x7f05016a; + public static final int invite_choice_title=0x7f05016f; + public static final int invite_chooser_email=0x7f050174; /** When I've created the invitation, in text or html, I ask Android to launch an app that can send it, typically an email or messaging app. Android then asks the user to choose which @@ -1353,38 +1363,38 @@ public final class R { to have launched. This string is passed to Android and used as the title of the dialog that presents that choice. */ - public static final int invite_chooser_fmt=0x7f050171; - public static final int invite_chooser_sms=0x7f050173; - public static final int invite_dict_missing_body_fmt=0x7f05023a; - public static final int invite_dict_missing_body_noname_fmt=0x7f05023b; - public static final int invite_dict_missing_title=0x7f050239; - public static final int invite_host=0x7f050078; + public static final int invite_chooser_fmt=0x7f050173; + public static final int invite_chooser_sms=0x7f050175; + public static final int invite_dict_missing_body_fmt=0x7f050241; + public static final int invite_dict_missing_body_noname_fmt=0x7f050242; + public static final int invite_dict_missing_title=0x7f050240; + public static final int invite_host=0x7f050079; /** This is the body of the html version of the invitation. A URL is created with parameters describing the game and substituted for "%1$s". (The funky \u003c and friends are encodings for the greater-than and less-than symbols which are not legal in xml strings.) */ - public static final int invite_htm_fmt=0x7f05016f; + public static final int invite_htm_fmt=0x7f050171; /** Appended to message above if local device has NFC available */ - public static final int invite_if_nfc=0x7f0500e0; - public static final int invite_mime=0x7f05007a; - public static final int invite_multi_summary=0x7f0502bc; - public static final int invite_multi_title=0x7f0502bb; + public static final int invite_if_nfc=0x7f0500e1; + public static final int invite_mime=0x7f05007b; + public static final int invite_multi_summary=0x7f0502c3; + public static final int invite_multi_title=0x7f0502c2; /** Most users create games with only two players, which is the default, but Crosswords supports up to four. When I'm using the above string to encourage the opener of a game missing players to invite, IF the number of missing players is greater than one this text is appended to the above. */ - public static final int invite_multiple=0x7f0500e1; - public static final int invite_prefix=0x7f050079; - public static final int invite_progress_fmt=0x7f0501e4; + public static final int invite_multiple=0x7f0500e2; + public static final int invite_prefix=0x7f05007a; + public static final int invite_progress_fmt=0x7f0501e6; /** */ - public static final int invite_progress_title=0x7f0501e3; - public static final int invite_stays=0x7f0502c3; + public static final int invite_progress_title=0x7f0501e5; + public static final int invite_stays=0x7f0502ca; /** Send invitation using SMS (texting) or via email? Send invitation using NFC (Android @@ -1400,25 +1410,25 @@ public final class R { This is the subject line of the email/text sent to invite someone to join a game. */ - public static final int invite_subject_fmt=0x7f05016e; + public static final int invite_subject_fmt=0x7f050170; /** This is the body of the text version of the invitation. A URL is created with parameters describing the game and substituted for "%1$s". */ - public static final int invite_txt_fmt=0x7f050170; + public static final int invite_txt_fmt=0x7f050172; /** text of checkbox asking if user wants to search for open public rooms */ - public static final int join_room=0x7f0500b6; + public static final int join_room=0x7f0500b7; /** Checkbox that when set keeps the device screen from dimming if the board screen is what's displayed. This is to allow users to think about a move without having to touch the screen all the time. */ - public static final int keep_screenon=0x7f050135; + public static final int keep_screenon=0x7f050136; /** clarification of above */ - public static final int keep_screenon_summary=0x7f050136; + public static final int keep_screenon_summary=0x7f050137; public static final int key_addrs_pref=0x7f05003e; public static final int key_background=0x7f05001a; public static final int key_board_size=0x7f050029; @@ -1429,7 +1439,7 @@ public final class R { /** the color of text, e.g. "2L", shown on a bonus square on the board */ - public static final int key_bonushint=0x7f050141; + public static final int key_bonushint=0x7f050142; public static final int key_bt_addrs=0x7f050040; /** database keys whose entries aren't visible prefs */ @@ -1440,31 +1450,31 @@ public final class R { */ public static final int key_color_tiles=0x7f050002; public static final int key_connect_frequency=0x7f050032; - public static final int key_connstat_data=0x7f050042; + public static final int key_connstat_data=0x7f050043; public static final int key_default_dict=0x7f05002c; - public static final int key_default_group=0x7f050047; + public static final int key_default_group=0x7f050048; public static final int key_default_language=0x7f05002b; public static final int key_default_loc=0x7f05003b; public static final int key_default_phonies=0x7f050030; public static final int key_default_robodict=0x7f05002f; public static final int key_default_timerenabled=0x7f050031; - public static final int key_dev_id=0x7f050043; + public static final int key_dev_id=0x7f050044; public static final int key_dict_host=0x7f050023; public static final int key_disable_nag=0x7f050007; - public static final int key_download_path=0x7f05006c; + public static final int key_download_path=0x7f05006d; public static final int key_empty=0x7f050019; - public static final int key_enable_debug=0x7f050067; - public static final int key_enable_dup_invite=0x7f050068; - public static final int key_enable_nfc_toself=0x7f050069; + public static final int key_enable_debug=0x7f050068; + public static final int key_enable_dup_invite=0x7f050069; + public static final int key_enable_nfc_toself=0x7f05006a; public static final int key_enable_pubroom=0x7f05000f; public static final int key_enable_sms=0x7f050035; - public static final int key_enable_sms_toself=0x7f05006a; + public static final int key_enable_sms_toself=0x7f05006b; public static final int key_explain_robot=0x7f050005; public static final int key_force_radio=0x7f05003d; public static final int key_force_tablet=0x7f05003c; - public static final int key_gcmvers_regid=0x7f050044; - public static final int key_got_langdict=0x7f05006d; - public static final int key_group_posns=0x7f050048; + public static final int key_gcmvers_regid=0x7f050045; + public static final int key_got_langdict=0x7f05006e; + public static final int key_group_posns=0x7f050049; public static final int key_hide_crosshairs=0x7f05000a; public static final int key_hide_newgames=0x7f05000e; public static final int key_hide_title=0x7f05000d; @@ -1473,46 +1483,46 @@ public final class R { public static final int key_init_hintsallowed=0x7f050026; public static final int key_init_nethintsallowed=0x7f050027; public static final int key_initial_player_minutes=0x7f05002a; - public static final int key_invite_multi=0x7f050070; + public static final int key_invite_multi=0x7f050071; public static final int key_keep_screenon=0x7f050037; - public static final int key_last_packet=0x7f050049; + public static final int key_last_packet=0x7f05004a; public static final int key_logging_on=0x7f050024; - public static final int key_na_browse=0x7f050061; - public static final int key_na_browseall=0x7f050062; - public static final int key_na_comms_bt=0x7f050072; - public static final int key_na_comms_relay=0x7f050074; - public static final int key_na_comms_sms=0x7f050073; - public static final int key_na_dicts=0x7f050066; - public static final int key_na_fmt_expl=0x7f050065; - public static final int key_na_lookup=0x7f050060; - public static final int key_na_studycopy=0x7f050064; - public static final int key_na_values=0x7f050063; - public static final int key_nag_intervals=0x7f05006b; + public static final int key_na_browse=0x7f050062; + public static final int key_na_browseall=0x7f050063; + public static final int key_na_comms_bt=0x7f050073; + public static final int key_na_comms_relay=0x7f050075; + public static final int key_na_comms_sms=0x7f050074; + public static final int key_na_dicts=0x7f050067; + public static final int key_na_fmt_expl=0x7f050066; + public static final int key_na_lookup=0x7f050061; + public static final int key_na_studycopy=0x7f050065; + public static final int key_na_values=0x7f050064; + public static final int key_nag_intervals=0x7f05006c; public static final int key_network_behavior=0x7f050036; - public static final int key_notagain_arrow=0x7f05005b; - public static final int key_notagain_backclears=0x7f05004d; - public static final int key_notagain_chat=0x7f05004e; - public static final int key_notagain_conndall=0x7f050058; - public static final int key_notagain_conndfirst=0x7f050059; - public static final int key_notagain_conndmid=0x7f05005a; - public static final int key_notagain_done=0x7f050056; - public static final int key_notagain_enablepublic=0x7f050071; - public static final int key_notagain_flip=0x7f050053; - public static final int key_notagain_hidenewgamebuttons=0x7f05005f; - public static final int key_notagain_hintnext=0x7f050051; - public static final int key_notagain_hintprev=0x7f050050; - public static final int key_notagain_juggle=0x7f050052; - public static final int key_notagain_newfrom=0x7f05005d; + public static final int key_notagain_arrow=0x7f05005c; + public static final int key_notagain_backclears=0x7f05004e; + public static final int key_notagain_chat=0x7f05004f; + public static final int key_notagain_conndall=0x7f050059; + public static final int key_notagain_conndfirst=0x7f05005a; + public static final int key_notagain_conndmid=0x7f05005b; + public static final int key_notagain_done=0x7f050057; + public static final int key_notagain_enablepublic=0x7f050072; + public static final int key_notagain_flip=0x7f050054; + public static final int key_notagain_hidenewgamebuttons=0x7f050060; + public static final int key_notagain_hintnext=0x7f050052; + public static final int key_notagain_hintprev=0x7f050051; + public static final int key_notagain_juggle=0x7f050053; + public static final int key_notagain_newfrom=0x7f05005e; /** key_notagain_sms_ready */ - public static final int key_notagain_newselect=0x7f05004c; - public static final int key_notagain_relay=0x7f05004f; - public static final int key_notagain_sync=0x7f05004b; - public static final int key_notagain_trading=0x7f05005e; - public static final int key_notagain_turnchanged=0x7f05005c; - public static final int key_notagain_undo=0x7f050055; - public static final int key_notagain_unlock=0x7f050057; - public static final int key_notagain_zoom=0x7f050054; + public static final int key_notagain_newselect=0x7f05004d; + public static final int key_notagain_relay=0x7f050050; + public static final int key_notagain_sync=0x7f05004c; + public static final int key_notagain_trading=0x7f05005f; + public static final int key_notagain_turnchanged=0x7f05005d; + public static final int key_notagain_undo=0x7f050056; + public static final int key_notagain_unlock=0x7f050058; + public static final int key_notagain_zoom=0x7f050055; public static final int key_notify_sound=0x7f050033; public static final int key_notify_vibrate=0x7f050034; public static final int key_peek_other=0x7f050009; @@ -1523,9 +1533,10 @@ public final class R { public static final int key_player3=0x7f050013; public static final int key_proxy_port=0x7f050021; public static final int key_relay_host=0x7f05001d; + public static final int key_relay_ids=0x7f050042; public static final int key_relay_port=0x7f05001e; - public static final int key_relay_regid=0x7f050045; - public static final int key_relay_regid_ackd=0x7f050046; + public static final int key_relay_regid=0x7f050046; + public static final int key_relay_regid_ackd=0x7f050047; public static final int key_ringer_zoom=0x7f05000c; public static final int key_robot_name=0x7f05002e; public static final int key_show_arrow=0x7f050003; @@ -1539,175 +1550,175 @@ public final class R { public static final int key_summary_field=0x7f05003a; public static final int key_thumbsize=0x7f050038; public static final int key_tile_back=0x7f050018; - public static final int key_udp_interval=0x7f05004a; + public static final int key_udp_interval=0x7f05004b; public static final int key_update_prerel=0x7f050020; public static final int key_update_url=0x7f05001f; - public static final int key_xlations_enabled=0x7f05006f; - public static final int key_xlations_locale=0x7f05006e; + public static final int key_xlations_enabled=0x7f050070; + public static final int key_xlations_locale=0x7f05006f; /** text of separator marking out the language area of the dialog. First is used for single-device games, and second for networked games (where players aren't allowed to have different wordlists.) */ - public static final int lang_label=0x7f0500b3; - public static final int lang_name_arabic=0x7f050283; - public static final int lang_name_catalan=0x7f05028a; - public static final int lang_name_czech=0x7f05028d; - public static final int lang_name_danish=0x7f050287; - public static final int lang_name_dutch=0x7f050289; - public static final int lang_name_english=0x7f05027f; - public static final int lang_name_french=0x7f050280; - public static final int lang_name_german=0x7f050281; - public static final int lang_name_greek=0x7f05028e; - public static final int lang_name_italian=0x7f050288; - public static final int lang_name_polish=0x7f050286; - public static final int lang_name_portuguese=0x7f05028b; - public static final int lang_name_russian=0x7f05028c; - public static final int lang_name_slovak=0x7f05028f; - public static final int lang_name_spanish=0x7f050284; - public static final int lang_name_swedish=0x7f050285; - public static final int lang_name_turkish=0x7f050282; - public static final int lang_unknown=0x7f0502a2; - public static final int langdict_label=0x7f0500b4; - public static final int list_group_default=0x7f050245; - public static final int list_group_delete=0x7f050243; - public static final int list_group_movedown=0x7f050247; - public static final int list_group_moveup=0x7f050246; - public static final int list_group_rename=0x7f050244; + public static final int lang_label=0x7f0500b4; + public static final int lang_name_arabic=0x7f05028a; + public static final int lang_name_catalan=0x7f050291; + public static final int lang_name_czech=0x7f050294; + public static final int lang_name_danish=0x7f05028e; + public static final int lang_name_dutch=0x7f050290; + public static final int lang_name_english=0x7f050286; + public static final int lang_name_french=0x7f050287; + public static final int lang_name_german=0x7f050288; + public static final int lang_name_greek=0x7f050295; + public static final int lang_name_italian=0x7f05028f; + public static final int lang_name_polish=0x7f05028d; + public static final int lang_name_portuguese=0x7f050292; + public static final int lang_name_russian=0x7f050293; + public static final int lang_name_slovak=0x7f050296; + public static final int lang_name_spanish=0x7f05028b; + public static final int lang_name_swedish=0x7f05028c; + public static final int lang_name_turkish=0x7f050289; + public static final int lang_unknown=0x7f0502a9; + public static final int langdict_label=0x7f0500b5; + public static final int list_group_default=0x7f05024c; + public static final int list_group_delete=0x7f05024a; + public static final int list_group_movedown=0x7f05024e; + public static final int list_group_moveup=0x7f05024d; + public static final int list_group_rename=0x7f05024b; /** ############## menu items ############## pulls up dialog to configure the selected game */ - public static final int list_item_config=0x7f050092; + public static final int list_item_config=0x7f050093; /** makes a copy of the selected game. This is currently disabled for networked games since there would be problems if two identically configured games started trying to talk to a remote game that expected there was only one of them. */ - public static final int list_item_copy=0x7f050098; + public static final int list_item_copy=0x7f050099; /** pulls up dialog to delete the selected game */ - public static final int list_item_delete=0x7f050095; + public static final int list_item_delete=0x7f050096; /** pulls up dialog to change the group of the selected game */ - public static final int list_item_move=0x7f050094; + public static final int list_item_move=0x7f050095; /** creates a new game with all configuation copied from the selected game */ - public static final int list_item_new_from=0x7f050097; + public static final int list_item_new_from=0x7f050098; /** pulls up dialog to rename (change name of) the selected game */ - public static final int list_item_rename=0x7f050093; + public static final int list_item_rename=0x7f050094; /** pulls up dialog to reset the selected game, that is to remove all moves so that it's the same as a newly created game except for any configuration. */ - public static final int list_item_reset=0x7f050096; + public static final int list_item_reset=0x7f050097; /** formatting for last move summary in notifications */ - public static final int lmi_pass_fmt=0x7f0502aa; - public static final int lmi_phony_fmt=0x7f0502ab; - public static final int lmi_tiles_fmt=0x7f0502ac; + public static final int lmi_pass_fmt=0x7f0502b1; + public static final int lmi_phony_fmt=0x7f0502b2; + public static final int lmi_tiles_fmt=0x7f0502b3; /** One of the strings used in the right column of the list of installed wordlists to describe those that are part of Crosswords and that cannot be uninstalled or moved. loc_internal and loc_external are the other possible strings in this column. */ - public static final int loc_builtin=0x7f05009f; + public static final int loc_builtin=0x7f0500a0; /** Used to describe wordlists that are in the Downloads directory. Currently I don't look there so this is unused, but I will eventually do so. This should be the same name as the built-in Android web browser uses for the directory where it saves files it downloads. */ - public static final int loc_downloads=0x7f0500a0; + public static final int loc_downloads=0x7f0500a1; /** */ - public static final int loc_external=0x7f0500aa; - public static final int loc_filters_all=0x7f05027b; - public static final int loc_filters_menu=0x7f05027d; - public static final int loc_filters_modified=0x7f05027e; - public static final int loc_filters_prompt=0x7f050279; - public static final int loc_filters_screen=0x7f05027c; - public static final int loc_fmts_mismatch=0x7f050294; + public static final int loc_external=0x7f0500ab; + public static final int loc_filters_all=0x7f050282; + public static final int loc_filters_menu=0x7f050284; + public static final int loc_filters_modified=0x7f050285; + public static final int loc_filters_prompt=0x7f050280; + public static final int loc_filters_screen=0x7f050283; + public static final int loc_fmts_mismatch=0x7f05029b; /** see move_dictf above */ - public static final int loc_internal=0x7f0500a9; - public static final int loc_item_check=0x7f050291; + public static final int loc_internal=0x7f0500aa; + public static final int loc_item_check=0x7f050298; /** for loc item edit menu */ - public static final int loc_item_clear=0x7f050290; - public static final int loc_item_copy_bless=0x7f050293; - public static final int loc_item_copy_eng=0x7f050292; - public static final int loc_lang_blessed=0x7f050274; - public static final int loc_lang_local=0x7f050275; - public static final int loc_menu_xlate=0x7f050273; - public static final int loc_search_prompt=0x7f05027a; + public static final int loc_item_clear=0x7f050297; + public static final int loc_item_copy_bless=0x7f05029a; + public static final int loc_item_copy_eng=0x7f050299; + public static final int loc_lang_blessed=0x7f05027b; + public static final int loc_lang_local=0x7f05027c; + public static final int loc_menu_xlate=0x7f05027a; + public static final int loc_search_prompt=0x7f050281; /** This is the "hint" printed in light text in the empty player name field */ - public static final int local_name_hint=0x7f050179; - public static final int logging_on=0x7f0502d3; - public static final int logging_on_summary=0x7f0502d4; - public static final int lookup_title=0x7f05026f; + public static final int local_name_hint=0x7f05017b; + public static final int logging_on=0x7f0502da; + public static final int logging_on_summary=0x7f0502db; + public static final int lookup_title=0x7f050276; /** */ - public static final int manual_owner_name=0x7f0501fd; + public static final int manual_owner_name=0x7f050201; /** */ - public static final int max_len=0x7f0501da; + public static final int max_len=0x7f0501dc; /** */ - public static final int menu_chat=0x7f05020d; + public static final int menu_chat=0x7f050214; /** */ - public static final int menu_flip=0x7f05020b; + public static final int menu_flip=0x7f050212; /** */ - public static final int menu_hint_next=0x7f050208; + public static final int menu_hint_next=0x7f05020f; /** */ - public static final int menu_hint_prev=0x7f050207; + public static final int menu_hint_prev=0x7f05020e; /** */ - public static final int menu_juggle=0x7f05020a; + public static final int menu_juggle=0x7f050211; /** text of menu that brings up the Settings (preferences) dialog */ - public static final int menu_prefs=0x7f05008c; - public static final int menu_rateme=0x7f050261; + public static final int menu_prefs=0x7f05008d; + public static final int menu_rateme=0x7f050268; /** */ - public static final int menu_toggle_values=0x7f05020e; + public static final int menu_toggle_values=0x7f050215; /** */ - public static final int menu_zoom=0x7f05020c; + public static final int menu_zoom=0x7f050213; /** */ - public static final int min_len=0x7f0501d9; + public static final int min_len=0x7f0501db; /** label for the field used to set the timer's inital value */ - public static final int minutes_label=0x7f0500c0; + public static final int minutes_label=0x7f0500c1; /** body of notification shown when invitation requires a wordslist that's not installed */ - public static final int missing_dict_detail=0x7f050238; + public static final int missing_dict_detail=0x7f05023f; /** Title of notification shown when invitation requires a wordslist that's not installed */ - public static final int missing_dict_title=0x7f050237; + public static final int missing_dict_title=0x7f05023e; /** Used as a substitute for the names of remote players when they aren't available yet because the connection is not complete. Displayed in the lists of players found in each game listing. */ - public static final int missing_player=0x7f050083; + public static final int missing_player=0x7f050084; /** When the dicts_item_move menu is chosen, this text is used in the confirmation dialog. The name of the selected wordlist is substituted for %1$s. The strings loc_internal and loc_internal are substitued for %2$s and %3$s (or vice-versa, depending on the current location of the wordlist.) */ - public static final int move_dict_fmt=0x7f0500a8; + public static final int move_dict_fmt=0x7f0500a9; /** ############################################################ # :Dialogs: @@ -1720,7 +1731,7 @@ public final class R { Text of dialog. Player name is substituted */ - public static final int msg_ask_password_fmt=0x7f050191; + public static final int msg_ask_password_fmt=0x7f050193; /** When a game has been connected and the relay is notified that a device in the game has deleted its part of the game, this message is posted when you connect your end of it to the @@ -1728,93 +1739,93 @@ public final class R { game so you might as well delete it (unless you're saving it for its history etc.) */ - public static final int msg_dev_deleted=0x7f0500e9; + public static final int msg_dev_deleted=0x7f0500ea; /** (I believe this can no longer occur) */ - public static final int msg_dup_room=0x7f0500e7; + public static final int msg_dup_room=0x7f0500e8; /** (I believe this can no longer occur) */ - public static final int msg_lost_other=0x7f0500e8; + public static final int msg_lost_other=0x7f0500e9; /** (I believe this can no longer occur) */ - public static final int msg_no_room=0x7f0500e6; + public static final int msg_no_room=0x7f0500e7; /** Text of "toast" shown when a game is notified by the relay that all expected players have registered. At this point play can begin. */ - public static final int msg_relay_all_here_fmt=0x7f0500e3; + public static final int msg_relay_all_here_fmt=0x7f0500e4; /** (I believe this can no longer occur) */ - public static final int msg_too_many=0x7f0500e5; + public static final int msg_too_many=0x7f0500e6; /** Text of progress indicator shown while check is being conducted */ - public static final int msgs_progress=0x7f05008e; - public static final int nag_body_fmt=0x7f0502a7; - public static final int nag_intervals=0x7f0502e4; + public static final int msgs_progress=0x7f05008f; + public static final int nag_body_fmt=0x7f0502ae; + public static final int nag_intervals=0x7f0502eb; /** Nagging: title of notification reminder message */ - public static final int nag_title=0x7f0502a6; + public static final int nag_title=0x7f0502ad; /** above is inserted in this the last time I warn */ - public static final int nag_warn_last_fmt=0x7f0502a8; - public static final int name_copy_fmt=0x7f0501f7; - public static final int name_dict_fmt=0x7f0502de; - public static final int nbs_port=0x7f05007d; + public static final int nag_warn_last_fmt=0x7f0502af; + public static final int name_copy_fmt=0x7f0501fa; + public static final int name_dict_fmt=0x7f0502e5; + public static final int nbs_port=0x7f05007e; /** text of checkbox. If this checkbox is checked, games created for network play will by default have the hint feature enabled. */ - public static final int nethints_allowed=0x7f0500be; + public static final int nethints_allowed=0x7f0500bf; /** clarification on hints_allowed, whether new NETWORKED games will default to having the hint feature enabled(string elsewhere in this file) */ - public static final int nethints_allowed_sum=0x7f050122; - public static final int netstats_title=0x7f0502da; - public static final int network_advanced_summary=0x7f0502ba; - public static final int network_advanced_title=0x7f0502b9; - public static final int network_behavior=0x7f050153; + public static final int nethints_allowed_sum=0x7f050123; + public static final int netstats_title=0x7f0502e1; + public static final int network_advanced_summary=0x7f0502c1; + public static final int network_advanced_title=0x7f0502c0; + public static final int network_behavior=0x7f050154; /** explanation of the above */ - public static final int network_behavior_summary=0x7f050154; + public static final int network_behavior_summary=0x7f050155; /** */ - public static final int new_app_avail=0x7f05022e; + public static final int new_app_avail=0x7f050235; /** */ - public static final int new_app_avail_fmt=0x7f05022d; + public static final int new_app_avail_fmt=0x7f050234; /** */ - public static final int new_bt_body_fmt=0x7f0501ea; + public static final int new_bt_body_fmt=0x7f0501ec; /** */ - public static final int new_bt_title=0x7f0501e9; + public static final int new_bt_title=0x7f0501eb; /** */ - public static final int new_btmove_title=0x7f0501ed; + public static final int new_btmove_title=0x7f0501ef; /** */ - public static final int new_dict_avail=0x7f05022b; + public static final int new_dict_avail=0x7f050232; /** */ - public static final int new_dict_avail_fmt=0x7f05022c; - public static final int new_game=0x7f0502b1; - public static final int new_game_message=0x7f0502b3; - public static final int new_game_message_net=0x7f0502b5; - public static final int new_game_message_nodflt=0x7f0502b4; - public static final int new_game_networked=0x7f0502b2; + public static final int new_dict_avail_fmt=0x7f050233; + public static final int new_game=0x7f0502b8; + public static final int new_game_message=0x7f0502ba; + public static final int new_game_message_net=0x7f0502bc; + public static final int new_game_message_nodflt=0x7f0502bb; + public static final int new_game_networked=0x7f0502b9; /** */ - public static final int new_move_body=0x7f0501ee; + public static final int new_move_body=0x7f0501f0; /** */ - public static final int new_name_body_fmt=0x7f0501fb; + public static final int new_name_body_fmt=0x7f0501fe; /** hint (text shown when field is empty) for room name field */ - public static final int new_room_hint=0x7f0500b7; + public static final int new_room_hint=0x7f0500b8; /** */ - public static final int new_sms_title=0x7f0501fa; + public static final int new_sms_title=0x7f0501fd; /** ########################################################### # :Dialogs: @@ -1827,33 +1838,33 @@ public final class R { Title of New user info dialog */ - public static final int newbie_title=0x7f0501a3; + public static final int newbie_title=0x7f0501a5; /** section separator (white-on-gray bar) for third section: bluetooth games */ - public static final int newgame_bt_header=0x7f050168; + public static final int newgame_bt_header=0x7f050169; /** Text of second of two buttons for new standalone games. Tap this and you'll get taken to the "Game configure" screen */ - public static final int newgame_configure_first=0x7f050163; - public static final int newgame_drop_relay=0x7f050167; + public static final int newgame_configure_first=0x7f050164; + public static final int newgame_drop_relay=0x7f050168; /** Text of first of two buttons for new networked games. Tap this and a game will be created, but you probably won't see it immediately because an email or messaging app will be launched to send your invitation. */ - public static final int newgame_invite=0x7f050166; + public static final int newgame_invite=0x7f050167; /** Text of first of two buttons for new standalone games. Tap this and a new game will be created and opened. If the first player is a robot it will immediately take its turn. */ - public static final int newgame_local=0x7f050162; + public static final int newgame_local=0x7f050163; /** This is one of two descriptions on this screen. It explains what standalone games are and describes the two buttons used to create them. The name of the language of the default wordlist is substituted in for %1$s. */ - public static final int newgame_local_desc_fmt=0x7f050161; + public static final int newgame_local_desc_fmt=0x7f050162; /** ############################################################ # :Screens: # New game screen @@ -1867,33 +1878,33 @@ public final class R { section separator (white-on-gray bar) for first section: standalone games */ - public static final int newgame_local_header=0x7f050160; + public static final int newgame_local_header=0x7f050161; /** This is the second of two descriptions on this screen. It explains what networked games are and describes the two buttons used to create them. */ - public static final int newgame_networked_desc=0x7f050165; + public static final int newgame_networked_desc=0x7f050166; /** section separator (white-on-gray bar) for second section: networked games */ - public static final int newgame_networked_header=0x7f050164; + public static final int newgame_networked_header=0x7f050165; /** */ - public static final int newgame_sms_header=0x7f0501f8; - public static final int newgroup_label=0x7f050242; - public static final int nfc_just_tap=0x7f0502c4; - public static final int nfc_to_self=0x7f0502ee; + public static final int newgame_sms_header=0x7f0501fb; + public static final int newgroup_label=0x7f050249; + public static final int nfc_just_tap=0x7f0502cb; + public static final int nfc_to_self=0x7f0502f5; /** Title of dialog for renaming game (triggered by selecting list_item_rename) If you try to copy a networked game you get this error message. */ - public static final int no_copy_network=0x7f050099; + public static final int no_copy_network=0x7f05009a; /** If the wordlist disappears mid-game there are no choices, just an explanation and this button, after which the game closes. */ - public static final int no_dict_finish=0x7f05018a; + public static final int no_dict_finish=0x7f05018c; /** If the missing wordlist is discovered when trying to open the game, we have more options. If there's another wordlist in the same language, we can offer to substitute without needing @@ -1902,12 +1913,12 @@ public final class R { takes wordlist name and language substituted in for %1$ and %2$ */ - public static final int no_dict_fmt=0x7f05018c; + public static final int no_dict_fmt=0x7f05018e; /** This is an alternative message presented when there's also the option of downloading another wordlist. Game name, wordlist name and language are substituted in. */ - public static final int no_dict_subst_fmt=0x7f05018d; + public static final int no_dict_subst_fmt=0x7f05018f; /** ############################################################ # :Dialogs: @@ -1919,26 +1930,26 @@ public final class R { title of alert */ - public static final int no_dict_title=0x7f050189; + public static final int no_dict_title=0x7f05018b; /** If you click on the Play button without having entered a room name you get an alert with this error message. */ - public static final int no_empty_rooms=0x7f05017b; + public static final int no_empty_rooms=0x7f05017d; /** If you choose the above option and have no networked games you get this error message */ - public static final int no_games_to_refresh=0x7f05008f; - public static final int no_market=0x7f050262; + public static final int no_games_to_refresh=0x7f050090; + public static final int no_market=0x7f050269; /** displayed when you long-tap a scoreboard entry and there's no most recent score to show */ - public static final int no_moves_made=0x7f0500df; + public static final int no_moves_made=0x7f0500e0; /** If the query returns no rooms, this message is displayed, with the number of players in the game and its language substituted. */ - public static final int no_name_found_fmt=0x7f0500bb; - public static final int no_relay_conn=0x7f050301; + public static final int no_name_found_fmt=0x7f0500bc; + public static final int no_relay_conn=0x7f050308; /** This is not currently shown Crosswords wordlists, which are just compressed lists of words plus tile information, determine @@ -1951,55 +1962,55 @@ public final class R { figuring out how to play and when you tap an empty cell the arrow appears. This explains it. */ - public static final int not_again_arrow=0x7f0501b6; - public static final int not_again_backclears=0x7f050255; + public static final int not_again_arrow=0x7f0501b8; + public static final int not_again_backclears=0x7f05025c; /** */ - public static final int not_again_browse=0x7f0501d6; + public static final int not_again_browse=0x7f0501d8; /** */ - public static final int not_again_browseall=0x7f0501d7; + public static final int not_again_browseall=0x7f0501d9; /** Shown when you tap the chat button on the toolbar of the main Board screen */ - public static final int not_again_chat=0x7f0501af; - public static final int not_again_comms_bt=0x7f05030a; - public static final int not_again_comms_relay=0x7f050308; - public static final int not_again_comms_sms=0x7f050309; + public static final int not_again_chat=0x7f0501b1; + public static final int not_again_comms_bt=0x7f050311; + public static final int not_again_comms_relay=0x7f05030f; + public static final int not_again_comms_sms=0x7f050310; /** This is shown in the Board screen when you successfully connecting a game to the relay and are the last device in the game to do so, i.e. the game is now complete and you should expect play to begin. */ - public static final int not_again_conndall=0x7f0501b5; + public static final int not_again_conndall=0x7f0501b7; /** This is shown in the Board screen when you successfully connect a game to the relay and are the first device in the game to do so. */ - public static final int not_again_conndfirst=0x7f0501b3; + public static final int not_again_conndfirst=0x7f0501b5; /** This is shown in the Board screen when you successfully connecting a game to the relay and are not the first device in the game but not the last either. So it will only occur for games with more than two devices, which are rare. */ - public static final int not_again_conndmid=0x7f0501b4; - public static final int not_again_dicts=0x7f0502a3; + public static final int not_again_conndmid=0x7f0501b6; + public static final int not_again_dicts=0x7f0502aa; /** This is shown when you choose the board_menu_done menu item. It's to let you know that there's a shortcut that does almost the same thing. */ - public static final int not_again_done=0x7f0501b1; - public static final int not_again_enablepublic=0x7f0502ca; + public static final int not_again_done=0x7f0501b3; + public static final int not_again_enablepublic=0x7f0502d1; /** Shown when you tap the flip button on the toolbar of the main Board screen */ - public static final int not_again_flip=0x7f0501ac; - public static final int not_again_fmt_expl=0x7f050295; - public static final int not_again_hidenewgamebuttons=0x7f0502c1; + public static final int not_again_flip=0x7f0501ae; + public static final int not_again_fmt_expl=0x7f05029c; + public static final int not_again_hidenewgamebuttons=0x7f0502c8; /** Shown when you tap the next hint button on the toolbar of the main Board screen */ - public static final int not_again_hintnext=0x7f0501aa; + public static final int not_again_hintnext=0x7f0501ac; /** Currently not used The new game you have created has two players. Player 1 is a robot; Player 2 is you. Tap the game @@ -2008,75 +2019,75 @@ public final class R { Shown when you tap the Previous Hint button on the toolbar of the main Board screen */ - public static final int not_again_hintprev=0x7f0501a9; + public static final int not_again_hintprev=0x7f0501ab; /** Shown when you tap the juggle button on the toolbar of the main Board screen */ - public static final int not_again_juggle=0x7f0501ab; + public static final int not_again_juggle=0x7f0501ad; /** */ - public static final int not_again_lookup=0x7f0501cc; + public static final int not_again_lookup=0x7f0501ce; /** Shown when you first pick the list_item_new_from menuitem */ - public static final int not_again_newfrom=0x7f0501b8; - public static final int not_again_newselect=0x7f050254; - public static final int not_again_studycopy=0x7f05026e; + public static final int not_again_newfrom=0x7f0501ba; + public static final int not_again_newselect=0x7f05025b; + public static final int not_again_studycopy=0x7f050275; /** The following strings (all whose names start with "not_again") appear in the New user info dialog. shown when user chooses the gamel_menu_checkmoves menu */ - public static final int not_again_sync=0x7f0501a5; + public static final int not_again_sync=0x7f0501a7; /** Shown when the user chooses the "board_menu_trade" menu */ - public static final int not_again_trading=0x7f0501a6; - public static final int not_again_trading_buttons=0x7f0501a7; - public static final int not_again_trading_menu=0x7f0501a8; + public static final int not_again_trading=0x7f0501a8; + public static final int not_again_trading_buttons=0x7f0501a9; + public static final int not_again_trading_menu=0x7f0501aa; /** Shown when the board screen is visible and it's just become another players turn. The idea is to give a hint about how to find out about recent moves. */ - public static final int not_again_turnchanged=0x7f0501b7; + public static final int not_again_turnchanged=0x7f0501b9; /** Shown when you tap the undo/redo button on the toolbar of the main Board screen */ - public static final int not_again_undo=0x7f0501ae; + public static final int not_again_undo=0x7f0501b0; /** Shown in the Game configure screen when the game_locked checkbox is checked and you uncheck it. */ - public static final int not_again_unlock=0x7f0501b2; + public static final int not_again_unlock=0x7f0501b4; /** Shown when you tap the values button on the toolbar of the main Board screen. This is intended to allow players to remind themselves how much played tiles are worth while planning a move. */ - public static final int not_again_values=0x7f0501b0; + public static final int not_again_values=0x7f0501b2; /** Shown when you tap the zoom (+/-) button on the toolbar of the main Board screen */ - public static final int not_again_zoom=0x7f0501ad; - public static final int note_none=0x7f05029f; + public static final int not_again_zoom=0x7f0501af; + public static final int note_none=0x7f0502a6; /** This text is displayed as the "summary" for both of the notify choices above, as extra information. (There could be different strings if it makes more sense in your language.) */ - public static final int notify_other_summary=0x7f05015f; + public static final int notify_other_summary=0x7f050160; /** When one or more new moves is found, should I play a notification sound */ - public static final int notify_sound=0x7f05015d; + public static final int notify_sound=0x7f05015e; /** When a move is fetched from the relay a Notification is posted. These are its title, which appears in the top bar of the device, and the body that appears when you pull the notifications down. */ - public static final int notify_title_fmt=0x7f050194; + public static final int notify_title_fmt=0x7f050196; /** When one or more new moves is found, should I vibrate the device */ - public static final int notify_vibrate=0x7f05015e; - public static final int nplayers_prompt=0x7f0502b7; + public static final int notify_vibrate=0x7f05015f; + public static final int nplayers_prompt=0x7f0502be; /** text of label identifying the field where human players can enter an option password. The label and field disappear when the robot player checkbox is checked because it makes no @@ -2085,13 +2096,13 @@ public final class R { player on a device, so they are infrequently used in network games as well. */ - public static final int password_label=0x7f0500d7; + public static final int password_label=0x7f0500d8; /** You have NFC enabled. That means that any time a board that\'s missing a player is open, you can tap a nearby person\'s device to invite him/her to play – if he/she is also using NFC. */ - public static final int pct_suffix=0x7f050260; + public static final int pct_suffix=0x7f050267; /** If this preference is checked, tapping on the scoreboard entry for any player reveals that player's tiles and any pending move (after asking for his password if one is set.) @@ -2101,41 +2112,41 @@ public final class R { though it's not his turn e.g. while the player whose turn it is is temporarily unable to play. */ - public static final int peek_other=0x7f05014f; + public static final int peek_other=0x7f050150; /** explanation of the above */ - public static final int peek_other_summary=0x7f050150; + public static final int peek_other_summary=0x7f050151; /** */ - public static final int phone_label=0x7f050203; + public static final int phone_label=0x7f05020a; /** Don't warn, but simply force to skip turn (give 0 points) when user attempts to play word not in the wordlist. */ - public static final int phonies_disallow=0x7f0500c8; + public static final int phonies_disallow=0x7f0500c9; /** These are the three choices in the popup above whose text is phonies_spinner_prompt Don't care if words played are in the wordlist or not */ - public static final int phonies_ignore=0x7f0500c6; + public static final int phonies_ignore=0x7f0500c7; /** title of popup used to determine how words are handled that are not in the wordlist used for the game (or player if using different wordlists per player) */ - public static final int phonies_spinner_prompt=0x7f0500c5; + public static final int phonies_spinner_prompt=0x7f0500c6; /** warn player when word played is not in the wordlist, but allow him to play it. */ - public static final int phonies_warn=0x7f0500c7; + public static final int phonies_warn=0x7f0500c8; /** */ - public static final int pick_faceup=0x7f0501d4; + public static final int pick_faceup=0x7f0501d6; /** */ - public static final int pick_url_title_fmt=0x7f0501ca; + public static final int pick_url_title_fmt=0x7f0501cc; /** Text of button at bottom. Press it and you exit this configure screen and open the game into the Board screen */ - public static final int play=0x7f0500cd; + public static final int play=0x7f0500ce; /** ############################################################ # :Dialogs: @@ -2147,47 +2158,47 @@ public final class R { dialog title */ - public static final int player_edit_title=0x7f0500d1; + public static final int player_edit_title=0x7f0500d2; /** used to create default player names. Number between 1 and 4 is substituted */ - public static final int player_fmt=0x7f050193; + public static final int player_fmt=0x7f050195; /** text of lable identifying the field in which player's name is set/displayed */ - public static final int player_label=0x7f0500d3; + public static final int player_label=0x7f0500d4; /** text for separator above the list of players that's used for networked games. The numbers of local and non-local players are substituted for %1$d and %2$d. */ - public static final int players_label_host_fmt=0x7f0500b0; + public static final int players_label_host_fmt=0x7f0500b1; /** text for separator above the list of players that's used for non-networked games */ - public static final int players_label_standalone=0x7f0500af; - public static final int pref_group_l10n_summary=0x7f0502fa; - public static final int pref_group_l10n_title=0x7f0502f9; - public static final int pref_group_relay_summary=0x7f0502f8; - public static final int pref_group_relay_title=0x7f0502f7; - public static final int pref_group_sms_summary=0x7f0502f6; - public static final int pref_group_sms_title=0x7f0502f5; + public static final int players_label_standalone=0x7f0500b0; + public static final int pref_group_l10n_summary=0x7f050301; + public static final int pref_group_l10n_title=0x7f050300; + public static final int pref_group_relay_summary=0x7f0502ff; + public static final int pref_group_relay_title=0x7f0502fe; + public static final int pref_group_sms_summary=0x7f0502fd; + public static final int pref_group_sms_title=0x7f0502fc; /** Label for the first "human player" name preference */ - public static final int pref_human_name=0x7f05011b; - public static final int pref_item_update_summary=0x7f0502fc; - public static final int pref_item_update_title=0x7f0502fb; + public static final int pref_human_name=0x7f05011c; + public static final int pref_item_update_summary=0x7f050303; + public static final int pref_item_update_title=0x7f050302; /** Label for the first player color preference */ - public static final int pref_player1_name=0x7f050117; + public static final int pref_player1_name=0x7f050118; /** Label for the second player color preference */ - public static final int pref_player2_name=0x7f050118; + public static final int pref_player2_name=0x7f050119; /** Label for the third player color preference */ - public static final int pref_player3_name=0x7f050119; + public static final int pref_player3_name=0x7f05011a; /** Label for the fourth player color preference */ - public static final int pref_player4_name=0x7f05011a; + public static final int pref_player4_name=0x7f05011b; /** ############################################################ # :Screens: @@ -2196,10 +2207,10 @@ public final class R { title of this sub-preference */ - public static final int prefs_appearance=0x7f050128; + public static final int prefs_appearance=0x7f050129; /** clarification of the above */ - public static final int prefs_appearance_summary=0x7f050129; + public static final int prefs_appearance_summary=0x7f05012a; /** ############################################################ # :Screens: @@ -2208,10 +2219,10 @@ public final class R { title of this sub-preference */ - public static final int prefs_behavior=0x7f050145; + public static final int prefs_behavior=0x7f050146; /** clarification of the above */ - public static final int prefs_behavior_summary=0x7f050146; + public static final int prefs_behavior_summary=0x7f050147; /** ############################################################ # :Screens: @@ -2221,10 +2232,10 @@ public final class R { title of this sub-preference */ - public static final int prefs_colors=0x7f050137; + public static final int prefs_colors=0x7f050138; /** clarification of the above */ - public static final int prefs_colors_summary=0x7f050138; + public static final int prefs_colors_summary=0x7f050139; /** ############################################################ # :Screens: @@ -2233,53 +2244,53 @@ public final class R { title of this sub-preference */ - public static final int prefs_defaults=0x7f050113; + public static final int prefs_defaults=0x7f050114; /** clarification of the above */ - public static final int prefs_defaults_summary=0x7f050114; + public static final int prefs_defaults_summary=0x7f050115; /** sub-preference for dictionaries (soon to be called "word lists") */ - public static final int prefs_dicts=0x7f05011d; + public static final int prefs_dicts=0x7f05011e; /** clarification of above */ - public static final int prefs_dicts_summary=0x7f05011e; + public static final int prefs_dicts_summary=0x7f05011f; /** sub-preference title for editing default player names. There's only enabled now, though. */ - public static final int prefs_names=0x7f050115; + public static final int prefs_names=0x7f050116; /** clarification of the above */ - public static final int prefs_names_summary=0x7f050116; + public static final int prefs_names_summary=0x7f050117; /** Used when prev player's name can't be looked up */ - public static final int prev_player=0x7f0502a9; - public static final int progress_title=0x7f05029d; + public static final int prev_player=0x7f0502b0; + public static final int progress_title=0x7f0502a4; /** */ - public static final int prompt_max_len=0x7f0501dc; + public static final int prompt_max_len=0x7f0501de; /** */ - public static final int prompt_min_len=0x7f0501db; - public static final int proxy_port=0x7f0502dd; + public static final int prompt_min_len=0x7f0501dd; + public static final int proxy_port=0x7f0502e4; /** Short for "points", this is shown at the right end of the tray in place of the first tile placed along with the points the current move would earn if committed. */ - public static final int pts=0x7f0500e2; + public static final int pts=0x7f0500e3; /** If you check the join_room checkbox Crosswords queries the relay. This is the text of the progress indicator displayed what that's going on. (It's often visible for only a very short time.) */ - public static final int public_names_progress_fmt=0x7f0500ba; + public static final int public_names_progress_fmt=0x7f0500bb; /** Title for generic dialog asking a question, usually in the middle of a game, like "do you want to commit this move?" */ - public static final int query_title=0x7f0501a2; - public static final int radio_name_cdma=0x7f0502ed; - public static final int radio_name_gsm=0x7f0502ec; - public static final int radio_name_real=0x7f0502ea; - public static final int radio_name_tablet=0x7f0502eb; + public static final int query_title=0x7f0501a4; + public static final int radio_name_cdma=0x7f0502f4; + public static final int radio_name_gsm=0x7f0502f3; + public static final int radio_name_real=0x7f0502f1; + public static final int radio_name_tablet=0x7f0502f2; /** ############################################################ # :Dialogs: @@ -2292,71 +2303,72 @@ public final class R { Names of the three colors by which colors can be edited in the color preferences dialog */ - public static final int red=0x7f050142; - public static final int redir_host=0x7f0502d1; + public static final int red=0x7f050143; + public static final int redir_host=0x7f0502d8; /** Title of dialog used to alert players to relay-related problems with the current game. */ - public static final int relay_alert=0x7f0500e4; - public static final int relay_alert_title=0x7f050253; - public static final int relay_host=0x7f0502d0; - public static final int relay_port=0x7f0502dc; + public static final int relay_alert=0x7f0500e5; + public static final int relay_alert_title=0x7f05025a; + public static final int relay_host=0x7f0502d7; + public static final int relay_invite_title=0x7f0501f6; + public static final int relay_port=0x7f0502e3; /** Shown in toast when relaunching after switching dicts */ - public static final int reload_new_dict_fmt=0x7f050236; - public static final int rematch_msg=0x7f0502c9; - public static final int remote_digesting=0x7f050297; - public static final int remote_empty=0x7f050296; + public static final int reload_new_dict_fmt=0x7f05023d; + public static final int rematch_msg=0x7f0502d0; + public static final int remote_digesting=0x7f05029e; + public static final int remote_empty=0x7f05029d; /** checkbox determining of this player is on this device or remote. If remote, then the rest of the fields disappear (since they will be set by the remote device.) */ - public static final int remote_label=0x7f0500d2; - public static final int remote_no_net=0x7f050298; + public static final int remote_label=0x7f0500d3; + public static final int remote_no_net=0x7f05029f; /** */ - public static final int remote_undone=0x7f050206; - public static final int rename_group_label=0x7f05024a; + public static final int remote_undone=0x7f05020d; + public static final int rename_group_label=0x7f050251; /** text within rename dialog (triggered by selecting list_item_rename) */ - public static final int rename_label=0x7f05009b; - public static final int rename_label_caveat=0x7f05009c; + public static final int rename_label=0x7f05009c; + public static final int rename_label_caveat=0x7f05009d; /** if this preference is checked, the hardware volume keys will work to zoom the board in and out (and will not control volume) This only applies when the Board screen is frontmost. */ - public static final int ringer_zoom=0x7f05014d; + public static final int ringer_zoom=0x7f05014e; /** explanation of the above */ - public static final int ringer_zoom_summary=0x7f05014e; + public static final int ringer_zoom_summary=0x7f05014f; /** checkbox determining if player is robot/automated or human */ - public static final int robot_label=0x7f0500d6; + public static final int robot_label=0x7f0500d7; /** Used to format robot player names in the lists of players found in each game listing */ - public static final int robot_name_fmt=0x7f050082; + public static final int robot_name_fmt=0x7f050083; /** Three possible choices presented in the popup above */ - public static final int robot_smart=0x7f0500c2; - public static final int robot_smarter=0x7f0500c3; - public static final int robot_smartest=0x7f0500c4; + public static final int robot_smart=0x7f0500c3; + public static final int robot_smarter=0x7f0500c4; + public static final int robot_smartest=0x7f0500c5; /** title of popup used to select how "smart" (how capable) the robot player will be. */ - public static final int robot_spinner_prompt=0x7f0500c1; + public static final int robot_spinner_prompt=0x7f0500c2; /** text of checkbox */ - public static final int room_public=0x7f0500b8; + public static final int room_public=0x7f0500b9; /** title for popup of public rooms found on server */ - public static final int room_public_prompt=0x7f0500b9; - public static final int seeking_relay=0x7f050300; - public static final int sel_games_fmt=0x7f050256; - public static final int sel_groups_fmt=0x7f050257; - public static final int sel_items_fmt=0x7f050272; + public static final int room_public_prompt=0x7f0500ba; + public static final int seeking_relay=0x7f050307; + public static final int sel_games_fmt=0x7f05025d; + public static final int sel_groups_fmt=0x7f05025e; + public static final int sel_items_fmt=0x7f050279; /** ############################################################ # :Dialogs: @@ -2367,59 +2379,59 @@ public final class R { up when the dicts_item_select menuitem is chosen. The possible answers are the three button text strings below. */ - public static final int set_default_message_fmt=0x7f0500a4; - public static final int set_pref=0x7f0502c0; + public static final int set_default_message_fmt=0x7f0500a5; + public static final int set_pref=0x7f0502c7; /** text of separator marking out other-setting area of the dialog */ - public static final int settings_label=0x7f0500bc; + public static final int settings_label=0x7f0500bd; /** Checkbox that when set makes taps on the board manipulate an arrow that then directs where tiles go when tapped in the tray. */ - public static final int show_arrow=0x7f050133; + public static final int show_arrow=0x7f050134; /** clarification of above */ - public static final int show_arrow_summary=0x7f050134; - public static final int show_remote=0x7f05029b; - public static final int show_wordlist_browser=0x7f050252; + public static final int show_arrow_summary=0x7f050135; + public static final int show_remote=0x7f0502a2; + public static final int show_wordlist_browser=0x7f050259; /** If this preference is checked, the user will not be asked to confirm after selecting the "Turn done" menu (or tapping the points display at the right end of the tray) */ - public static final int skip_confirm_turn=0x7f050149; + public static final int skip_confirm_turn=0x7f05014a; /** explanation of the above */ - public static final int skip_confirm_turn_summary=0x7f05014a; - public static final int slmenu_clear_sel=0x7f050268; - public static final int slmenu_copy_sel=0x7f050267; - public static final int slmenu_deselect_all=0x7f050271; - public static final int slmenu_select_all=0x7f050270; - public static final int sms_bad_proto_fmt=0x7f0501ec; + public static final int skip_confirm_turn_summary=0x7f05014b; + public static final int slmenu_clear_sel=0x7f05026f; + public static final int slmenu_copy_sel=0x7f05026e; + public static final int slmenu_deselect_all=0x7f050278; + public static final int slmenu_select_all=0x7f050277; + public static final int sms_bad_proto_fmt=0x7f0501ee; /** Title of phone number picker during invitation to a game via SMS */ - public static final int sms_invite_title=0x7f0501f3; - public static final int sms_ready_text=0x7f05025f; - public static final int square_tiles=0x7f05024f; - public static final int square_tiles_summary=0x7f050250; + public static final int sms_invite_title=0x7f0501f5; + public static final int sms_ready_text=0x7f050266; + public static final int square_tiles=0x7f050256; + public static final int square_tiles_summary=0x7f050257; /** Used in formatting moves and history */ - public static final int str_bonus_all=0x7f05010b; + public static final int str_bonus_all=0x7f05010c; /** Using the hint feature is cheating by some players, and it can be disabled via the hints_allowed preference. I should be disabling the buttons in this case, but if I don't and you try to get a hint you'll get this message instead. */ - public static final int str_cant_hint_while_disabled=0x7f050185; + public static final int str_cant_hint_while_disabled=0x7f050187; /** Displayed if you try to use the undo menuitem or button and there are no tiles on the board (no move has yet been made.) [If I'm being clever and disabling those features in this case there may be no way to see this.] */ - public static final int str_cant_undo_tileassign=0x7f050184; + public static final int str_cant_undo_tileassign=0x7f050186; /** Beginning of the message presented to a user when asking him to confirm committing the current turn */ - public static final int str_commit_confirm=0x7f050108; + public static final int str_commit_confirm=0x7f050109; /** Used to format game name plus some other information as the one-line summary for each game in the main screen. The name of the game is substituted for %1$s. Something else @@ -2427,16 +2439,16 @@ public final class R { does not require translation unless the parentheses or ordering is wrong for your language. */ - public static final int str_game_name_fmt=0x7f050081; + public static final int str_game_name_fmt=0x7f050082; /** Displayed when you try to commit a turn that is illegal because there is empty space between some of the tiles placed, i.e. they do not form a single word. */ - public static final int str_no_empties_in_turn=0x7f05017d; + public static final int str_no_empties_in_turn=0x7f05017f; /** Same as above, but used when you try to show tiles belonging to a player on another device (a remote player.) */ - public static final int str_no_peek_remote_tiles=0x7f050182; + public static final int str_no_peek_remote_tiles=0x7f050184; /** Displayed when you try to reveal a robot player's tiles, either by tapping on its "hidden" rack (marked by "?" characters) or by tapping on its scoreboard entry. (The @@ -2444,28 +2456,28 @@ public final class R { controlled by the peek_other preference and is disabled by default.) */ - public static final int str_no_peek_robot_tiles=0x7f050181; + public static final int str_no_peek_robot_tiles=0x7f050183; /** Used, with remote player's name substituted for %1$s, to indicate that the player is remote. */ - public static final int str_nonlocal_name_fmt=0x7f050109; + public static final int str_nonlocal_name_fmt=0x7f05010a; /** Displyed when you try to commit a move and it's not your turn. */ - public static final int str_not_your_turn=0x7f050180; + public static final int str_not_your_turn=0x7f050182; /** Used in formatting game history and move summaries */ - public static final int str_pass=0x7f050102; + public static final int str_pass=0x7f050103; /** Used to alert user to loss of turn when a move is made and phonies is set to lose turn when word used not in wordlist */ - public static final int str_phony_rejected=0x7f050107; - public static final int str_placer_fmt=0x7f050231; + public static final int str_phony_rejected=0x7f050108; + public static final int str_placer_fmt=0x7f050238; /** Shown when using the the Game configure screen to configure a networked game and you try to make all players local. */ - public static final int str_reg_server_sans_remote=0x7f0500c9; + public static final int str_reg_server_sans_remote=0x7f0500ca; /** This error message is shown when a remote device tries to join a game and is providing more players than that game expects. I do not believe it is possible to see this message @@ -2473,11 +2485,11 @@ public final class R { many players are expected and only connects devices where the numbers match. */ - public static final int str_reg_unexpected_user=0x7f050186; + public static final int str_reg_unexpected_user=0x7f050188; /** Used in formatting remote player move summaries */ - public static final int str_remote_moved_fmt=0x7f050100; - public static final int str_resigned_fmt=0x7f050230; + public static final int str_remote_moved_fmt=0x7f050101; + public static final int str_resigned_fmt=0x7f050237; /** ############################################################ # Board info/error dialog messages @@ -2488,18 +2500,18 @@ public final class R { Notifies user of a normal robot move */ - public static final int str_robot_moved_fmt=0x7f0500fa; + public static final int str_robot_moved_fmt=0x7f0500fb; /** Shown when a remote device tries to join a game and wants to play in a different language. As with the above, this should be impossible when connecting via the relay. */ - public static final int str_server_dict_wins=0x7f050187; + public static final int str_server_dict_wins=0x7f050189; /** Displayed when you try to commit a turn that is illegal because tiles played do not touch other tiles already on the board (or the middle/star square in the case of the initial move.) */ - public static final int str_tiles_must_contact=0x7f05017f; + public static final int str_tiles_must_contact=0x7f050181; /** error messages @@ -2507,71 +2519,71 @@ public final class R { because the tiles placed are not all in the same row or column. */ - public static final int str_tiles_not_in_line=0x7f05017c; + public static final int str_tiles_not_in_line=0x7f05017e; /** Displayed when you try to begin a trade but there are not seven or more tiles in the pool. The rules don't allow trading in this case. */ - public static final int str_too_few_tiles_left_to_trade=0x7f050183; + public static final int str_too_few_tiles_left_to_trade=0x7f050185; /** Displayed when you try to commit the first move of the game and are playing a single tile. The first move must include two or more tiles. */ - public static final int str_two_tiles_first_move=0x7f05017e; + public static final int str_two_tiles_first_move=0x7f050180; /** Used in formatting final scores display */ - public static final int str_winner_fmt=0x7f05022f; + public static final int str_winner_fmt=0x7f050236; /** Used in formatting game history (not move summaries since information about the current rack is hidden then) */ - public static final int strd_cumulative_score_fmt=0x7f0500ff; + public static final int strd_cumulative_score_fmt=0x7f050100; /** Used in formatting game history and move summaries */ - public static final int strd_remaining_tiles_add_fmt=0x7f0500fc; + public static final int strd_remaining_tiles_add_fmt=0x7f0500fd; /** I don't know how this is used. :-) */ - public static final int strd_time_penalty_sub_fmt=0x7f050101; + public static final int strd_time_penalty_sub_fmt=0x7f050102; /** Used in formatting moves and history. The total score for one turn is substituted for %1$d. */ - public static final int strd_turn_score_fmt=0x7f05010c; + public static final int strd_turn_score_fmt=0x7f05010d; /** Used in formatting game history and move summaries */ - public static final int strd_unused_tiles_sub_fmt=0x7f0500fd; + public static final int strd_unused_tiles_sub_fmt=0x7f0500fe; /** Used in formatting game history and move summaries */ - public static final int strs_move_across_fmt=0x7f050103; + public static final int strs_move_across_fmt=0x7f050104; /** Used in formatting game history and move summaries */ - public static final int strs_move_down_fmt=0x7f050104; + public static final int strs_move_down_fmt=0x7f050105; /** Used in formatting game history and move summaries */ - public static final int strs_new_tiles_fmt=0x7f0500fe; + public static final int strs_new_tiles_fmt=0x7f0500ff; /** Used in formatting game history and move summaries */ - public static final int strs_tray_at_start_fmt=0x7f050105; + public static final int strs_tray_at_start_fmt=0x7f050106; /** title for window you get when you select menu with text board_menu_game_counts */ - public static final int strs_values_header_fmt=0x7f0500fb; + public static final int strs_values_header_fmt=0x7f0500fc; /** Used in formatting exchange move summaries */ - public static final int strss_traded_for_fmt=0x7f050106; - public static final int study_langpick=0x7f05026b; - public static final int study_no_lang_fmt=0x7f05026c; - public static final int study_no_lists=0x7f05026d; - public static final int studylist_title_fmt=0x7f05026a; + public static final int strss_traded_for_fmt=0x7f050107; + public static final int study_langpick=0x7f050272; + public static final int study_no_lang_fmt=0x7f050273; + public static final int study_no_lists=0x7f050274; + public static final int studylist_title_fmt=0x7f050271; /** If substituting an existing same-language wordlist by choosing button_substdict user gets to choose from a list of wordlists. This is the title of that list. */ - public static final int subst_dict_title=0x7f050190; + public static final int subst_dict_title=0x7f050192; /** */ - public static final int summary_conn=0x7f0501e8; + public static final int summary_conn=0x7f0501ea; /** */ - public static final int summary_conn_sms_fmt=0x7f050204; + public static final int summary_conn_sms_fmt=0x7f05020b; /** Preference to control what's listed next to game name in the first line of a game summary in the main games-list screen, e.g., if the option is "Game Language", "English" in the @@ -2580,10 +2592,10 @@ public final class R { The following strings beginning with "game_summary_field_" are possible values */ - public static final int summary_field=0x7f05012a; + public static final int summary_field=0x7f05012b; /** */ - public static final int summary_gameover=0x7f0501e7; + public static final int summary_gameover=0x7f0501e9; /** The display of each networked game includes one of three states it can be in in the process of connecting to the relay. These next three strings are used to create those @@ -2591,57 +2603,57 @@ public final class R { First state: is configured to use a room but has not yet contacted the relay and been assigned that room. */ - public static final int summary_relay_conf_fmt=0x7f050084; + public static final int summary_relay_conf_fmt=0x7f050085; /** Third state: enough devices have connected in the room to form a complete game. We'll be in this state as long as the game exists. */ - public static final int summary_relay_conn_fmt=0x7f050086; + public static final int summary_relay_conn_fmt=0x7f050087; /** Final state: game is over. */ - public static final int summary_relay_gameover_fmt=0x7f050087; + public static final int summary_relay_gameover_fmt=0x7f050088; /** Second state: has been assigned to a room on the relay (meaning the network is working) but there are not yet as many players as expected: the game is not complete and play cannot happen. */ - public static final int summary_relay_wait_fmt=0x7f050085; - public static final int summary_send_data_sms=0x7f0502d8; + public static final int summary_relay_wait_fmt=0x7f050086; + public static final int summary_send_data_sms=0x7f0502df; /** explanation of the above */ - public static final int summary_sort_tiles=0x7f05014c; - public static final int summary_studyon=0x7f050265; - public static final int summary_thumbsize=0x7f050258; + public static final int summary_sort_tiles=0x7f05014d; + public static final int summary_studyon=0x7f05026c; + public static final int summary_thumbsize=0x7f05025f; /** */ - public static final int summary_wait_guest=0x7f0501e6; + public static final int summary_wait_guest=0x7f0501e8; /** */ - public static final int summary_wait_host=0x7f0501e5; + public static final int summary_wait_host=0x7f0501e7; /** Used to indicate that a preference is not enabled, i.e. not part of the game and that the user should ignore it. */ - public static final int tell_unused=0x7f05011c; - public static final int thumb_off=0x7f050259; + public static final int tell_unused=0x7f05011d; + public static final int thumb_off=0x7f050260; /** color of the tiles' background */ - public static final int tile_back=0x7f05013e; + public static final int tile_back=0x7f05013f; /** */ - public static final int tilepick_all=0x7f0501d2; + public static final int tilepick_all=0x7f0501d4; /** */ - public static final int tilepick_undo=0x7f0501d1; + public static final int tilepick_undo=0x7f0501d3; /** title of dialog brought up in response to the board_menu_game_left menu. The dialog lists all tiles remaining in the pool, i.e. not on the board or in the rack of the player whose turn it is. */ - public static final int tiles_left_title=0x7f05019d; + public static final int tiles_left_title=0x7f05019f; /** Title of preference in which you select which addressing modes network games will use to communicate */ - public static final int title_addrs_pref=0x7f0502b0; + public static final int title_addrs_pref=0x7f0502b7; /** ############################################################ # :Screens: @@ -2652,7 +2664,7 @@ public final class R { window title */ - public static final int title_dicts_list=0x7f05009d; + public static final int title_dicts_list=0x7f05009e; /** ############################################################ # :Screens: @@ -2661,17 +2673,17 @@ public final class R { window title (game name substituted for %1$s) */ - public static final int title_game_config_fmt=0x7f0500ab; + public static final int title_game_config_fmt=0x7f0500ac; /** alternate window title used when game is networked */ - public static final int title_gamenet_config_fmt=0x7f0500ac; + public static final int title_gamenet_config_fmt=0x7f0500ad; /** title for popup list of langugages from which user picks */ - public static final int title_langs_list=0x7f0500ad; + public static final int title_langs_list=0x7f0500ae; /** */ - public static final int title_lookup=0x7f0501c6; - public static final int title_lookup_study=0x7f0501c7; + public static final int title_lookup=0x7f0501c8; + public static final int title_lookup_study=0x7f0501c9; /** ############################################################ # :Screens: @@ -2684,52 +2696,52 @@ public final class R { window title */ - public static final int title_prefs=0x7f05010e; - public static final int title_send_data_sms=0x7f0502d7; + public static final int title_prefs=0x7f05010f; + public static final int title_send_data_sms=0x7f0502de; /** If this preference is checked, tiles in the rack will be re-ordered alphabetically whenever tiles are added, i.e. after ever move. */ - public static final int title_sort_tiles=0x7f05014b; - public static final int title_studyon=0x7f050264; + public static final int title_sort_tiles=0x7f05014c; + public static final int title_studyon=0x7f05026b; /** title of dialog allowing user to pick tiles "face up". (This feature is not yet supported on Android.) */ - public static final int title_tile_picker=0x7f05019c; - public static final int update_dicts_fmt=0x7f05029c; - public static final int use_defaults=0x7f0502b6; + public static final int title_tile_picker=0x7f05019e; + public static final int update_dicts_fmt=0x7f0502a3; + public static final int use_defaults=0x7f0502bd; /** text of checkbox controlling whether there's a game timer */ - public static final int use_timer=0x7f0500bf; + public static final int use_timer=0x7f0500c0; /** Used to separate names of players when listing them on one line in a game summary. The \u0020 is a space in xml. */ - public static final int vs_join=0x7f05010a; - public static final int waiting_title=0x7f0502c2; - public static final int warn_bt_disabled=0x7f050225; - public static final int warn_no_comms=0x7f0502c5; + public static final int vs_join=0x7f05010b; + public static final int waiting_title=0x7f0502c9; + public static final int warn_bt_disabled=0x7f05022c; + public static final int warn_no_comms=0x7f0502cc; /** */ - public static final int warn_nomobile_fmt=0x7f0501fe; + public static final int warn_nomobile_fmt=0x7f050202; /** */ - public static final int warn_sms_disabled=0x7f050224; + public static final int warn_sms_disabled=0x7f05022b; /** */ - public static final int warn_unlimited=0x7f050205; - public static final int wifi_warning=0x7f050304; + public static final int warn_unlimited=0x7f05020c; + public static final int wifi_warning=0x7f05030b; /** */ - public static final int word_search_hint=0x7f0501d0; - public static final int xlations_enabled_summary=0x7f050277; - public static final int xlations_enabled_title=0x7f050276; - public static final int xlations_locale=0x7f0502e2; + public static final int word_search_hint=0x7f0501d2; + public static final int xlations_enabled_summary=0x7f05027e; + public static final int xlations_enabled_title=0x7f05027d; + public static final int xlations_locale=0x7f0502e9; /** Empty in English, this should contain the name of the translator/creator of the strings.xml file for this language */ - public static final int xlator=0x7f0501be; - public static final int xwords_nfc_mime=0x7f050077; + public static final int xlator=0x7f0501c0; + public static final int xwords_nfc_mime=0x7f050078; } public static final class style { public static final int about_items=0x7f080005; diff --git a/xwords4/android/XWords4/jni/Android.mk b/xwords4/android/XWords4/jni/Android.mk index 9f02d5721..c37ba7dc2 100644 --- a/xwords4/android/XWords4/jni/Android.mk +++ b/xwords4/android/XWords4/jni/Android.mk @@ -39,6 +39,7 @@ LOCAL_DEFINES += \ -DXWFEATURE_BASE64 \ -DXWFEATURE_DEVID \ -DCOMMON_LAYOUT \ + -DNATIVE_NLI \ -DCOMMS_VERSION=1 \ -DINITIAL_CLIENT_VERS=${INITIAL_CLIENT_VERS} \ -DRELAY_ROOM_DEFAULT=\"\" \ @@ -83,7 +84,7 @@ COMMON_SRC_FILES += \ $(COMMON_PATH)/memstream.c \ $(COMMON_PATH)/movestak.c \ $(COMMON_PATH)/dbgutil.c \ - + $(COMMON_PATH)/invit.c \ LOCAL_CFLAGS+=$(LOCAL_C_INCLUDES) $(LOCAL_DEFINES) -Wall -std=c99 LOCAL_SRC_FILES := $(linux_SRC_FILES) $(LOCAL_SRC_FILES) $(COMMON_SRC_FILES) diff --git a/xwords4/android/XWords4/jni/andutils.c b/xwords4/android/XWords4/jni/andutils.c index ec2c9cbe6..496cfb8f7 100644 --- a/xwords4/android/XWords4/jni/andutils.c +++ b/xwords4/android/XWords4/jni/andutils.c @@ -78,12 +78,14 @@ error error error int getInt( JNIEnv* env, jobject obj, const char* name ) { + // XP_LOGF( "%s(name=%s)", __func__, name ); jclass cls = (*env)->GetObjectClass( env, obj ); XP_ASSERT( !!cls ); jfieldID fid = (*env)->GetFieldID( env, cls, name, "I"); XP_ASSERT( !!fid ); int result = (*env)->GetIntField( env, obj, fid ); deleteLocalRef( env, cls ); + // XP_LOGF( "%s(name=%s) => %d", __func__, name, result ); return result; } @@ -113,12 +115,14 @@ getInts( JNIEnv* env, void* cobj, jobject jobj, const SetInfo* sis, XP_U16 nSis void setInt( JNIEnv* env, jobject obj, const char* name, int value ) { + // XP_LOGF( "%s(name=%s)", __func__, name ); jclass cls = (*env)->GetObjectClass( env, obj ); XP_ASSERT( !!cls ); jfieldID fid = (*env)->GetFieldID( env, cls, name, "I"); XP_ASSERT( !!fid ); (*env)->SetIntField( env, obj, fid, value ); deleteLocalRef( env, cls ); + // XP_LOGF( "%s(name=%s) DONE", __func__, name ); } void @@ -178,6 +182,7 @@ setBools( JNIEnv* env, jobject jobj, void* cobj, const SetInfo* sis, XP_U16 nSis bool setString( JNIEnv* env, jobject obj, const char* name, const XP_UCHAR* value ) { + // XP_LOGF( "%s(%s)", __func__, name ); bool success = false; jclass cls = (*env)->GetObjectClass( env, obj ); jfieldID fid = (*env)->GetFieldID( env, cls, name, "Ljava/lang/String;" ); @@ -193,10 +198,32 @@ setString( JNIEnv* env, jobject obj, const char* name, const XP_UCHAR* value ) return success; } +void +getStrings( JNIEnv* env, void* cobj, jobject jobj, const SetInfo* sis, XP_U16 nSis ) +{ + for ( int ii = 0; ii < nSis; ++ii ) { + const SetInfo* si = &sis[ii]; + XP_UCHAR* buf = (XP_UCHAR*)(((uint8_t*)cobj) + si->offset); + getString( env, jobj, si->name, buf, si->siz ); + } +} + +void +setStrings( JNIEnv* env, jobject jobj, void* cobj, const SetInfo* sis, XP_U16 nSis ) +{ + for ( int ii = 0; ii < nSis; ++ii ) { + const SetInfo* si = &sis[ii]; + // XP_LOGF( "calling setString(%s)", si->name ); + XP_UCHAR* val = (XP_UCHAR*)(((uint8_t*)cobj) + si->offset); + setString( env, jobj, si->name, val ); + } +} + void getString( JNIEnv* env, jobject obj, const char* name, XP_UCHAR* buf, int bufLen ) { + // XP_LOGF( "%s(%s)", __func__, name ); jclass cls = (*env)->GetObjectClass( env, obj ); XP_ASSERT( !!cls ); jfieldID fid = (*env)->GetFieldID( env, cls, name, "Ljava/lang/String;" ); @@ -214,6 +241,7 @@ getString( JNIEnv* env, jobject obj, const char* name, XP_UCHAR* buf, buf[len] = '\0'; deleteLocalRef( env, cls ); + // XP_LOGF( "%s(%s) DONE", __func__, name ); } XP_UCHAR* diff --git a/xwords4/android/XWords4/jni/andutils.h b/xwords4/android/XWords4/jni/andutils.h index 214c41022..30e79dce9 100644 --- a/xwords4/android/XWords4/jni/andutils.h +++ b/xwords4/android/XWords4/jni/andutils.h @@ -58,6 +58,10 @@ void setBools( JNIEnv* env, jobject jobj, void* cobj, bool setString( JNIEnv* env, jobject obj, const char* name, const XP_UCHAR* value ); void getString( JNIEnv* env, jobject jlp, const char* name, XP_UCHAR* buf, int bufLen ); +void getStrings( JNIEnv* env, void* cobj, jobject jobj, + const SetInfo* sis, XP_U16 nSis ); +void setStrings( JNIEnv* env, jobject jobj, void* cobj, + const SetInfo* sis, XP_U16 nSis ); XP_UCHAR* getStringCopy( MPFORMAL JNIEnv* env, jstring jname ); void setObject( JNIEnv* env, jobject obj, const char* name, const char* sig, jobject val ); diff --git a/xwords4/android/XWords4/jni/xwjni.c b/xwords4/android/XWords4/jni/xwjni.c index 29ae3ea43..468ee7406 100644 --- a/xwords4/android/XWords4/jni/xwjni.c +++ b/xwords4/android/XWords4/jni/xwjni.c @@ -32,6 +32,7 @@ #include "dictnry.h" #include "dictiter.h" #include "dictmgr.h" +#include "invit.h" #include "utilwrapper.h" #include "drawwrapper.h" @@ -283,6 +284,46 @@ makeGI( MPFORMAL JNIEnv* env, jobject jgi ) return gi; } /* makeGI */ +static const SetInfo nli_ints[] = { + ARR_MEMBER( NetLaunchInfo, _conTypes ), + ARR_MEMBER( NetLaunchInfo, lang ), + ARR_MEMBER( NetLaunchInfo, forceChannel ), + ARR_MEMBER( NetLaunchInfo, nPlayersT ), + ARR_MEMBER( NetLaunchInfo, nPlayersH ), + ARR_MEMBER( NetLaunchInfo, gameID ), + ARR_MEMBER( NetLaunchInfo, osVers ), +}; + +static const SetInfo nli_bools[] = { + ARR_MEMBER( NetLaunchInfo, isGSM ) +}; + +static const SetInfo nli_strs[] = { + ARR_MEMBER( NetLaunchInfo, dict ), + ARR_MEMBER( NetLaunchInfo, gameName ), + ARR_MEMBER( NetLaunchInfo, room ), + ARR_MEMBER( NetLaunchInfo, btName ), + ARR_MEMBER( NetLaunchInfo, btAddress ), + ARR_MEMBER( NetLaunchInfo, phone ), + ARR_MEMBER( NetLaunchInfo, inviteID ), +}; + +static void +loadNLI( JNIEnv* env, NetLaunchInfo* nli, jobject jnli ) +{ + getInts( env, (void*)nli, jnli, nli_ints, VSIZE(nli_ints) ); + getBools( env, (void*)nli, jnli, nli_bools, VSIZE(nli_bools) ); + getStrings( env, (void*)nli, jnli, nli_strs, VSIZE(nli_strs) ); +} + +static void +setNLI( JNIEnv* env, jobject jnli, const NetLaunchInfo* nli ) +{ + setInts( env, jnli, (void*)nli, nli_ints, VSIZE(nli_ints) ); + setBools( env, jnli, (void*)nli, nli_bools, VSIZE(nli_bools) ); + setStrings( env, jnli, (void*)nli, nli_strs, VSIZE(nli_strs) ); +} + static void setJGI( JNIEnv* env, jobject jgi, const CurGameInfo* gi ) { @@ -453,14 +494,67 @@ Java_org_eehouse_android_xw4_jni_XwJNI_gi_1from_1stream #endif } +JNIEXPORT jbyteArray JNICALL +Java_org_eehouse_android_xw4_jni_XwJNI_nli_1to_1stream +( JNIEnv* env, jclass C, jobject njli ) +{ + LOG_FUNC(); + jbyteArray result; +#ifdef MEM_DEBUG + MemPoolCtx* mpool = mpool_make( NULL ); +#endif + NetLaunchInfo nli = {0}; + loadNLI( env, &nli, njli ); + /* CurGameInfo* gi = makeGI( MPPARM(mpool) env, jgi ); */ + VTableMgr* vtMgr = make_vtablemgr( MPPARM_NOCOMMA(mpool) ); + XWStreamCtxt* stream = mem_stream_make( MPPARM(mpool) vtMgr, + NULL, 0, NULL ); + + invit_saveToStream( &nli, stream ); + + result = streamToBArray( env, stream ); + stream_destroy( stream ); + + vtmgr_destroy( MPPARM(mpool) vtMgr ); +#ifdef MEM_DEBUG + mpool_destroy( mpool ); +#endif + return result; +} + +JNIEXPORT void JNICALL +Java_org_eehouse_android_xw4_jni_XwJNI_nli_1from_1stream +( JNIEnv* env, jclass C, jobject jnli, jbyteArray jstream ) +{ + LOG_FUNC(); +#ifdef MEM_DEBUG + MemPoolCtx* mpool = mpool_make( NULL ); +#endif + VTableMgr* vtMgr = make_vtablemgr( MPPARM_NOCOMMA(mpool) ); + XWStreamCtxt* stream = streamFromJStream( MPPARM(mpool) env, vtMgr, jstream ); + + NetLaunchInfo nli = {0}; + if ( invit_makeFromStream( &nli, stream ) ) { + setNLI( env, jnli, &nli ); + } else { + XP_LOGF( "%s: game_makeFromStream failed", __func__ ); + } + + stream_destroy( stream ); + vtmgr_destroy( MPPARM(mpool) vtMgr ); +#ifdef MEM_DEBUG + mpool_destroy( mpool ); +#endif +} + JNIEXPORT void JNICALL Java_org_eehouse_android_xw4_jni_XwJNI_comms_1getInitialAddr -( JNIEnv* env, jclass C, jobject jaddr, jstring jname, jint port ) +( JNIEnv* env, jclass C, jobject jaddr, jstring jname, jint port, jint devID ) { CommsAddrRec addr; const char* chars = (*env)->GetStringUTFChars( env, jname, NULL ); - comms_getInitialAddr( &addr, chars, port ); + comms_getInitialAddr( &addr, chars, port, devID ); (*env)->ReleaseStringUTFChars( env, jname, chars ); setJAddrRec( env, jaddr, &addr ); } diff --git a/xwords4/android/XWords4/res/layout/relayinviter.xml b/xwords4/android/XWords4/res/layout/relayinviter.xml new file mode 100644 index 000000000..95c1a1169 --- /dev/null +++ b/xwords4/android/XWords4/res/layout/relayinviter.xml @@ -0,0 +1,58 @@ + + + + + + + + + + +