mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2024-12-30 10:26:58 +01:00
78dfed6c59
Call it when turn changes, new game is begun, and hint feature has been used. Currently menuitem is duplicated. Goal is to remove the item being attached to the left button and to replace it when another is chosen.
3185 lines
94 KiB
C
Executable file
3185 lines
94 KiB
C
Executable file
/* -*- fill-column: 77; c-basic-offset: 4; compile-command: "make TARGET_OS=wince DEBUG=TRUE" -*- */
|
|
/*
|
|
* Copyright 2002-2007 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.
|
|
*
|
|
* Derived from code generated by M$'s eVC++.
|
|
*/
|
|
|
|
#include "stdafx.h"
|
|
#include "xwords4.h"
|
|
#include <commctrl.h>
|
|
#include <commdlg.h>
|
|
#include <stdio.h>
|
|
#include <time.h> /* time() */
|
|
#include <winuser.h>
|
|
#ifdef _WIN32_WCE
|
|
# include <aygshell.h>
|
|
#endif
|
|
#include "strutils.h"
|
|
|
|
#include "memstream.h"
|
|
|
|
#include "cemain.h"
|
|
#include "cedefines.h"
|
|
|
|
#include "ceginfo.h"
|
|
#include "cestrbx.h"
|
|
#include "cedict.h"
|
|
#include "ceblank.h"
|
|
#include "ceprefs.h"
|
|
#include "ceaskpwd.h"
|
|
#include "ceutil.h"
|
|
#include "ceir.h"
|
|
#include "ceclrsel.h"
|
|
#include "cehntlim.h"
|
|
#include "cedebug.h"
|
|
#include "LocalizedStrIncludes.h"
|
|
#include "debhacks.h"
|
|
|
|
#include "dbgutil.h"
|
|
|
|
#define MAX_LOADSTRING 100
|
|
|
|
#ifdef _WIN32_WCE
|
|
# define DEFAULT_DIR_NAME L"\\My Documents\\Crosswords"
|
|
# define PREFSFILENAME L"\\My Documents\\Crosswords\\xwprefs"
|
|
# define UNSAVEDGAMEFILENAME "\\My Documents\\Crosswords\\_newgame"
|
|
#else
|
|
# define DEFAULT_DIR_NAME L"."
|
|
# define PREFSFILENAME L".\\xwprefs"
|
|
# define UNSAVEDGAMEFILENAME ".\\_newgame"
|
|
#endif
|
|
|
|
#define SCROLLBAR_WIDTH 12
|
|
#define SCROLLBARID 0x4321 /* needs to be unique! */
|
|
|
|
#ifdef MEM_DEBUG
|
|
# define MEMPOOL globals->mpool,
|
|
#else
|
|
# define MEMPOOL
|
|
#endif
|
|
|
|
typedef struct FileWriteState {
|
|
CEAppGlobals* globals;
|
|
XP_UCHAR* path;
|
|
} FileWriteState;
|
|
|
|
/* forward util function decls */
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
static XP_S16 ce_send_proc( const XP_U8* buf, XP_U16 len,
|
|
const CommsAddrRec* addr,
|
|
void* closure );
|
|
|
|
#define CE_SEND_PROC ce_send_proc
|
|
#else
|
|
#define CE_SEND_PROC NULL
|
|
#endif
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
static void ce_reset_proc( void* closure );
|
|
# define CE_RESET_PROC ce_send_proc,
|
|
#else
|
|
# define CE_RESET_PROC
|
|
#endif
|
|
|
|
static VTableMgr* ce_util_getVTManager( XW_UtilCtxt* uc );
|
|
static void ce_util_userError( XW_UtilCtxt* uc, UtilErrID id );
|
|
static XP_Bool ce_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id,
|
|
XWStreamCtxt* stream );
|
|
static XWBonusType ce_util_getSquareBonus( XW_UtilCtxt* uc,
|
|
ModelCtxt* model,
|
|
XP_U16 col, XP_U16 row );
|
|
static XP_S16 ce_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* pi,
|
|
XP_U16 playerNum,
|
|
const XP_UCHAR4* texts, XP_U16 nTiles );
|
|
static XP_Bool ce_util_askPassword( XW_UtilCtxt* uc, const XP_UCHAR* name,
|
|
XP_UCHAR* buf, XP_U16* len );
|
|
static void ce_util_trayHiddenChange( XW_UtilCtxt* uc,
|
|
XW_TrayVisState newState,
|
|
XP_U16 nVisibleRows );
|
|
static void ce_util_yOffsetChange( XW_UtilCtxt* uc, XP_U16 oldOffset,
|
|
XP_U16 newOffset );
|
|
static void ce_util_turnChanged( XW_UtilCtxt* uc );
|
|
static void ce_util_notifyGameOver( XW_UtilCtxt* uc );
|
|
static XP_Bool ce_util_hiliteCell( XW_UtilCtxt* uc, XP_U16 col,
|
|
XP_U16 row );
|
|
static XP_Bool ce_util_engineProgressCallback( XW_UtilCtxt* uc );
|
|
static void ce_util_setTimer( XW_UtilCtxt* uc, XWTimerReason why, XP_U16 when,
|
|
XWTimerProc proc, void* closure);
|
|
static void ce_util_requestTime( XW_UtilCtxt* uc );
|
|
static XP_U32 ce_util_getCurSeconds( XW_UtilCtxt* uc );
|
|
static DictionaryCtxt* ce_util_makeEmptyDict( XW_UtilCtxt* uc );
|
|
#ifdef XWFEATURE_RELAY
|
|
static XWStreamCtxt* ce_util_makeStreamFromAddr( XW_UtilCtxt* uc,
|
|
XP_PlayerAddr channelNo );
|
|
#endif
|
|
static const XP_UCHAR* ce_util_getUserString( XW_UtilCtxt* uc,
|
|
XP_U16 stringCode );
|
|
static XP_Bool ce_util_warnIllegalWord( XW_UtilCtxt* uc, BadWordInfo* bwi,
|
|
XP_U16 turn, XP_Bool turnLost );
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_RELAY
|
|
static void ce_util_addrChange( XW_UtilCtxt* uc, const CommsAddrRec* oldAddr,
|
|
const CommsAddrRec* newAddr );
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static XP_Bool ce_util_getTraySearchLimits( XW_UtilCtxt* uc, XP_U16* min,
|
|
XP_U16* max );
|
|
#endif
|
|
#ifdef SHOW_PROGRESS
|
|
static void ce_util_engineStarting( XW_UtilCtxt* uc );
|
|
static void ce_util_engineStopping( XW_UtilCtxt* uc );
|
|
#endif
|
|
|
|
static void messageBoxChar( CEAppGlobals* globals, XP_UCHAR* str,
|
|
wchar_t* title );
|
|
static XP_Bool queryBoxChar( CEAppGlobals* globals, XP_UCHAR* msg );
|
|
|
|
static XP_Bool ceMsgFromStream( CEAppGlobals* globals, XWStreamCtxt* stream,
|
|
wchar_t* title, XP_Bool isQuery,
|
|
XP_Bool destroy );
|
|
static void RECTtoXPR( XP_Rect* dest, RECT* src );
|
|
static XP_Bool ceDoNewGame( CEAppGlobals* globals );
|
|
static XP_Bool ceSaveCurGame( CEAppGlobals* globals, XP_Bool autoSave );
|
|
static void ceInitPrefs( CEAppGlobals* globals, CEAppPrefs* prefs );
|
|
static void updateForColors( CEAppGlobals* globals );
|
|
static XWStreamCtxt* make_generic_stream( CEAppGlobals* globals );
|
|
#ifdef XWFEATURE_RELAY
|
|
static void ce_send_on_close( XWStreamCtxt* stream, void* closure );
|
|
#endif
|
|
static XP_Bool ceSetDictName( const wchar_t* wPath, XP_U16 index, void* ctxt );
|
|
static void messageBoxStream( CEAppGlobals* globals, XWStreamCtxt* stream,
|
|
wchar_t* title );
|
|
static XP_Bool ceQueryFromStream( CEAppGlobals* globals, XWStreamCtxt* stream);
|
|
#ifdef _WIN32_WCE
|
|
static void sizeIfFullscreen( CEAppGlobals* globals );
|
|
#endif
|
|
|
|
|
|
#if defined DEBUG && ! defined _WIN32_WCE
|
|
/* Very basic cmdline args meant at first to let me vary the size of the
|
|
* screen. Form is of arg=digits, with no spaces and digits having to be an
|
|
* integer. Right now only width and height work: e.g.
|
|
* "wine obj_win32_dbg/xwords4.exe height=240 width=320"
|
|
*/
|
|
|
|
static int g_dbWidth = 0;
|
|
static int g_dbHeight = 0;
|
|
|
|
static void
|
|
doCmd( const char* cmd )
|
|
{
|
|
struct { char* p; int* v; } params[] = {
|
|
{ "width", &g_dbWidth },
|
|
{ "height", &g_dbHeight }
|
|
};
|
|
int i;
|
|
|
|
for ( i = 0; i < VSIZE(params); ++i ) {
|
|
char* p = params[i].p;
|
|
int len = strlen(p);
|
|
if ( 0 == strncmp( p, cmd, len ) ) {
|
|
cmd += len;
|
|
if ( *cmd == '=' ) {
|
|
++cmd;
|
|
*(params[i].v) = atoi(cmd);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( i == VSIZE(params) ) {
|
|
XP_LOGF( "failed to match cmdline arg \"%s\"", cmd );
|
|
}
|
|
} /* doCmd */
|
|
|
|
static void
|
|
parseCmdLine( const char* cmdline )
|
|
{
|
|
for ( ; ; ) {
|
|
const char* cmd;
|
|
char ch;
|
|
char buf[64];
|
|
int len;
|
|
for ( cmd = cmdline ; ; ++cmd ) {
|
|
ch = *cmd;
|
|
if ( ch == '\0' || ch == ' ' ) {
|
|
break;
|
|
}
|
|
}
|
|
len = cmd - cmdline;
|
|
memcpy( buf, cmdline, cmd - cmdline );
|
|
buf[len] = '\0';
|
|
doCmd( buf );
|
|
if ( ch == '\0' ) {
|
|
break;
|
|
}
|
|
cmdline = ++cmd;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Forward declarations of functions included in this code module:
|
|
ATOM MyRegisterClass (HINSTANCE, LPTSTR);
|
|
BOOL InitInstance (HINSTANCE, int);
|
|
LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM);
|
|
LRESULT CALLBACK ceAbout (HWND, UINT, WPARAM, LPARAM);
|
|
|
|
int WINAPI
|
|
WinMain( HINSTANCE hInstance,
|
|
HINSTANCE XP_UNUSED(hPrevInstance),
|
|
#if defined TARGET_OS_WINCE
|
|
LPWSTR XP_UNUSED_CE(lpCmdLine),
|
|
#elif defined TARGET_OS_WIN32
|
|
LPSTR XP_UNUSED_DBG(lpCmdLine),
|
|
#endif
|
|
int nCmdShow)
|
|
{
|
|
MSG msg;
|
|
HACCEL hAccelTable;
|
|
|
|
#if defined DEBUG && ! defined _WIN32_WCE
|
|
parseCmdLine( lpCmdLine );
|
|
#endif
|
|
|
|
// Perform application initialization:
|
|
if (!InitInstance (hInstance, nCmdShow)) {
|
|
return FALSE;
|
|
}
|
|
|
|
hAccelTable = LoadAccelerators(hInstance, (LPCTSTR)IDC_XWORDS4);
|
|
|
|
// Main message loop. Return of 0 indicates quit message. Return of -1
|
|
// indicates major error (so we just bail.)
|
|
while ( 0 < GetMessage(&msg, NULL, 0, 0) ) {
|
|
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) {
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
}
|
|
|
|
/* This would be a good place to free up memory, close sockets, etc. */
|
|
|
|
LOG_RETURNF( "%d", msg.wParam );
|
|
return msg.wParam;
|
|
}
|
|
|
|
#ifdef __GNUC__
|
|
int
|
|
main()
|
|
{
|
|
XP_LOGF( "" );
|
|
LOG_FUNC();
|
|
|
|
return WinMain( GetModuleHandle(NULL), 0,
|
|
#if defined TARGET_OS_WINCE
|
|
GetCommandLineW(),
|
|
#elif defined TARGET_OS_WIN32
|
|
GetCommandLineA(),
|
|
#endif
|
|
SW_SHOWDEFAULT );
|
|
}
|
|
#endif
|
|
|
|
|
|
//
|
|
// FUNCTION: MyRegisterClass()
|
|
//
|
|
// PURPOSE: Registers the window class.
|
|
//
|
|
// COMMENTS:
|
|
//
|
|
// It is important to call this function so that the application
|
|
// will get 'well formed' small icons associated with it.
|
|
//
|
|
ATOM
|
|
MyRegisterClass(HINSTANCE hInstance, LPTSTR szWindowClass)
|
|
{
|
|
WNDCLASS wc;
|
|
|
|
XP_MEMSET( &wc, 0, sizeof(wc) );
|
|
|
|
wc.style = CS_HREDRAW | CS_VREDRAW;
|
|
wc.lpfnWndProc = (WNDPROC) WndProc;
|
|
wc.cbWndExtra = sizeof(CEAppGlobals*);
|
|
wc.hInstance = hInstance;
|
|
wc.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_XWORDS4));
|
|
wc.hbrBackground = (HBRUSH) GetStockObject(WHITE_BRUSH);
|
|
#if defined TARGET_OS_WIN32
|
|
wc.lpszMenuName = (LPCTSTR)IDM_MENU;
|
|
#endif
|
|
wc.lpszClassName = szWindowClass;
|
|
|
|
return RegisterClass(&wc);
|
|
}
|
|
|
|
static void
|
|
ceInitUtilFuncs( CEAppGlobals* globals )
|
|
{
|
|
UtilVtable* vtable = globals->util.vtable =
|
|
XP_MALLOC( globals->mpool, sizeof( UtilVtable ) );
|
|
globals->util.closure = (void*)globals;
|
|
globals->util.gameInfo = &globals->gameInfo;
|
|
|
|
MPASSIGN( globals->util.mpool, globals->mpool );
|
|
|
|
vtable->m_util_getVTManager = ce_util_getVTManager;
|
|
vtable->m_util_userError = ce_util_userError;
|
|
vtable->m_util_getSquareBonus = ce_util_getSquareBonus;
|
|
vtable->m_util_userQuery = ce_util_userQuery;
|
|
vtable->m_util_userPickTile = ce_util_userPickTile;
|
|
vtable->m_util_askPassword = ce_util_askPassword;
|
|
vtable->m_util_trayHiddenChange = ce_util_trayHiddenChange;
|
|
vtable->m_util_yOffsetChange = ce_util_yOffsetChange;
|
|
vtable->m_util_turnChanged = ce_util_turnChanged;
|
|
vtable->m_util_notifyGameOver = ce_util_notifyGameOver;
|
|
vtable->m_util_hiliteCell = ce_util_hiliteCell;
|
|
vtable->m_util_engineProgressCallback = ce_util_engineProgressCallback;
|
|
vtable->m_util_setTimer = ce_util_setTimer;
|
|
vtable->m_util_requestTime = ce_util_requestTime;
|
|
vtable->m_util_getCurSeconds = ce_util_getCurSeconds;
|
|
vtable->m_util_makeEmptyDict = ce_util_makeEmptyDict;
|
|
vtable->m_util_getUserString = ce_util_getUserString;
|
|
vtable->m_util_warnIllegalWord = ce_util_warnIllegalWord;
|
|
#ifdef XWFEATURE_RELAY
|
|
vtable->m_util_addrChange = ce_util_addrChange;
|
|
vtable->m_util_makeStreamFromAddr = ce_util_makeStreamFromAddr;
|
|
#endif
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
vtable->m_util_getTraySearchLimits = ce_util_getTraySearchLimits;
|
|
#endif
|
|
#ifdef SHOW_PROGRESS
|
|
vtable->m_util_engineStarting = ce_util_engineStarting;
|
|
vtable->m_util_engineStopping = ce_util_engineStopping;
|
|
#endif
|
|
|
|
} /* ceInitUtilFuncs */
|
|
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
# define SCROLL_SHRINK 1
|
|
|
|
static void
|
|
updateScrollInfo( HWND hwnd, XP_U16 nHidden )
|
|
{
|
|
SCROLLINFO sinfo;
|
|
|
|
XP_MEMSET( &sinfo, 0, sizeof(sinfo) );
|
|
sinfo.cbSize = sizeof(sinfo);
|
|
sinfo.fMask = SIF_RANGE | SIF_POS | SIF_PAGE;
|
|
sinfo.nPos = 0;
|
|
sinfo.nMin = 0;
|
|
sinfo.nMax = nHidden;
|
|
sinfo.nPage = 1;
|
|
|
|
(void)SetScrollInfo( hwnd, SB_CTL, &sinfo, TRUE );
|
|
}
|
|
|
|
static void
|
|
showScroller( CEAppGlobals* globals, XP_U16 nHidden, XP_U16 x, XP_U16 y,
|
|
XP_U16 width, XP_U16 height )
|
|
{
|
|
if ( !!globals->scrollHandle ) {
|
|
DestroyWindow( globals->scrollHandle );
|
|
globals->scrollHandle = NULL;
|
|
}
|
|
|
|
if ( !globals->scrollHandle ) {
|
|
HWND hwndSB;
|
|
|
|
hwndSB = CreateWindow( TEXT("SCROLLBAR"), // Class name
|
|
NULL, // Window text
|
|
// Window style
|
|
SBS_VERT|WS_VISIBLE|WS_CHILD,
|
|
x + SCROLL_SHRINK, y,
|
|
width - SCROLL_SHRINK, height + 1,
|
|
globals->hWnd,
|
|
(HMENU)SCROLLBARID,// The control identifier
|
|
globals->hInst, // The instance handle
|
|
NULL ); // s'pposed to be NULL
|
|
|
|
updateScrollInfo( hwndSB, nHidden );
|
|
|
|
EnableWindow( hwndSB, nHidden > 0 );
|
|
|
|
globals->scrollHandle = hwndSB;
|
|
}
|
|
|
|
ShowWindow( globals->scrollHandle, SW_SHOW );
|
|
} /* showScroller */
|
|
|
|
static void
|
|
hideScroller( CEAppGlobals* globals )
|
|
{
|
|
if ( !!globals->scrollHandle ) {
|
|
ShowWindow( globals->scrollHandle, SW_HIDE );
|
|
}
|
|
/* else there's nothing to do */
|
|
}
|
|
#endif
|
|
|
|
typedef struct CEBoardParms {
|
|
XP_U16 boardHScale;
|
|
XP_U16 boardVScale;
|
|
XP_U16 boardTop;
|
|
XP_U16 trayTop;
|
|
|
|
XP_U16 trayHeight;
|
|
XP_U16 trayWidth;
|
|
|
|
XP_U16 timerLeft, timerTop, timerWidth, timerHeight;
|
|
|
|
XP_U16 boardLeft, trayLeft;
|
|
XP_U16 scoreWidth;
|
|
XP_U16 scoreHeight;
|
|
XP_Bool needsScroller;
|
|
XP_Bool horiz;
|
|
} CEBoardParms;
|
|
|
|
static XP_U16
|
|
sizeBoard( XP_U16* bdHeightP, /* INOUT */
|
|
XP_U16* nRowsP, /* INOUT: on OUT, gives nRowsVisible */
|
|
XP_U16* scrollWidthP )
|
|
{
|
|
/* given the initial max board height, figure how many rows are visible
|
|
and the adjusted heights of the board and tray. */
|
|
XP_U16 bdHeight = *bdHeightP;
|
|
XP_U16 nVisibleRows = *nRowsP;
|
|
XP_U16 vScale;
|
|
XP_U16 boardHtLimit;
|
|
|
|
*scrollWidthP = 0;
|
|
|
|
vScale = bdHeight / nVisibleRows;
|
|
if ( vScale < MIN_CELL_HEIGHT ) {
|
|
vScale = MIN_CELL_HEIGHT;
|
|
}
|
|
|
|
/* Now adjust tray height to make board height a multiple */
|
|
boardHtLimit = nVisibleRows * vScale;
|
|
while ( boardHtLimit > bdHeight ) {
|
|
boardHtLimit -= vScale;
|
|
--nVisibleRows;
|
|
*scrollWidthP = SCROLLBAR_WIDTH;
|
|
}
|
|
|
|
*bdHeightP = boardHtLimit;
|
|
*nRowsP = nVisibleRows;
|
|
return vScale;
|
|
} /* sizeBoard */
|
|
|
|
static void
|
|
figureBoardParms( CEAppGlobals* globals, XP_U16 nRows, CEBoardParms* bparms )
|
|
{
|
|
RECT rc;
|
|
XP_U16 scrnWidth, scrnHeight;
|
|
XP_U16 trayVScale, boardLeft, scoreWidth, scoreHeight;
|
|
XP_U16 boardHt, boardWidth, hScale, vScale, nVisibleRows;
|
|
XP_U16 trayTop, boardTop;
|
|
XP_Bool horiz;
|
|
XP_U16 trayWidth;
|
|
XP_U16 scrollWidth = 0;
|
|
|
|
GetClientRect( globals->hWnd, &rc );
|
|
#ifndef _WIN32_WCE
|
|
# if defined FORCE_HEIGHT && defined FORCE_WIDTH
|
|
rc.right = rc.left + FORCE_WIDTH;
|
|
rc.bottom = rc.top + FORCE_HEIGHT;
|
|
# else
|
|
# if defined DEBUG
|
|
|
|
if ( g_dbWidth != 0 ) {
|
|
rc.right = rc.left + g_dbWidth;
|
|
}
|
|
if ( g_dbHeight != 0 ) {
|
|
rc.bottom = rc.top + g_dbHeight;
|
|
}
|
|
# endif
|
|
# endif
|
|
#endif /* #ifndef _WIN32_WCE */
|
|
|
|
scrnWidth = (XP_U16)(rc.right - rc.left);
|
|
scrnHeight = (XP_U16)(rc.bottom - rc.top);
|
|
|
|
horiz = (scrnHeight - CE_SCORE_HEIGHT) >= (scrnWidth - CE_MIN_SCORE_WIDTH);
|
|
nVisibleRows = nRows;
|
|
|
|
scoreHeight = horiz? CE_SCORE_HEIGHT : 0;
|
|
boardTop = scoreHeight;
|
|
|
|
/* Try to make it fit without scrolling. But if necessary, reduce the
|
|
width for a scrollbar. */
|
|
boardHt = scrnHeight - scoreHeight - MIN_TRAY_HEIGHT;
|
|
vScale = sizeBoard( &boardHt, &nVisibleRows, &scrollWidth );
|
|
|
|
boardWidth = scrnWidth - scrollWidth;
|
|
if ( horiz ) {
|
|
scoreWidth = scrnWidth;
|
|
hScale = boardWidth / nRows;
|
|
boardWidth = nRows * hScale;
|
|
/* center the board */
|
|
boardWidth += scrollWidth;
|
|
boardLeft = (scrnWidth - boardWidth) / 2; /* center it all */
|
|
} else {
|
|
/* move extra pixels into scoreboard */
|
|
hScale = (boardWidth - CE_MIN_SCORE_WIDTH) / nRows;
|
|
boardWidth = hScale * nRows;
|
|
scoreWidth = scrnWidth - boardWidth - scrollWidth;
|
|
boardLeft = scoreWidth;
|
|
}
|
|
trayWidth = boardWidth + scrollWidth;
|
|
|
|
trayTop = boardHt + scoreHeight + 1;
|
|
trayVScale = scrnHeight - trayTop;
|
|
|
|
if ( !horiz ) {
|
|
scoreHeight = scrnHeight;
|
|
}
|
|
|
|
if ( globals->gameInfo.timerEnabled ) {
|
|
if ( horiz ) {
|
|
scoreWidth -= CE_TIMER_WIDTH;
|
|
bparms->timerLeft = scoreWidth;
|
|
bparms->timerTop = 0;
|
|
bparms->timerWidth = CE_TIMER_WIDTH;
|
|
bparms->timerHeight = CE_SCORE_HEIGHT;
|
|
} else {
|
|
bparms->timerLeft = 0;
|
|
bparms->timerHeight = CE_SCORE_HEIGHT * 2;
|
|
bparms->timerTop = scrnHeight - bparms->timerHeight;
|
|
bparms->timerWidth = scoreWidth;
|
|
|
|
scoreHeight -= bparms->timerHeight;
|
|
}
|
|
}
|
|
|
|
bparms->boardHScale = hScale;
|
|
bparms->boardVScale = vScale;
|
|
bparms->boardTop = boardTop;
|
|
bparms->trayTop = trayTop;
|
|
bparms->trayHeight = trayVScale;
|
|
bparms->trayWidth = trayWidth;
|
|
bparms->boardLeft = boardLeft;
|
|
bparms->trayLeft = boardLeft;
|
|
bparms->scoreWidth = scoreWidth;
|
|
bparms->scoreHeight = scoreHeight;
|
|
bparms->horiz = horiz;
|
|
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
globals->nHiddenRows = nRows - nVisibleRows;
|
|
bparms->needsScroller = nVisibleRows < nRows;
|
|
if ( bparms->needsScroller ) {
|
|
XP_U16 boardRight = boardLeft + (nRows * hScale);
|
|
showScroller( globals, globals->nHiddenRows,
|
|
boardRight, boardTop,
|
|
scrollWidth, boardHt );
|
|
XP_LOGF( "NEEDING SCROLLBAR!!!!" );
|
|
XP_LOGF( "%d rows hidden", globals->nHiddenRows );
|
|
} else {
|
|
hideScroller( globals );
|
|
}
|
|
#endif
|
|
} /* figureBoardParms */
|
|
|
|
static XP_Bool
|
|
cePositionBoard( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool erase = XP_FALSE;
|
|
XP_U16 nCols;
|
|
CEBoardParms bparms;
|
|
|
|
XP_ASSERT( !!globals->game.model );
|
|
nCols = model_numCols( globals->game.model );
|
|
XP_ASSERT( nCols <= CE_MAX_ROWS );
|
|
|
|
figureBoardParms( globals, nCols, &bparms );
|
|
|
|
if ( globals->gameInfo.timerEnabled ) {
|
|
board_setTimerLoc( globals->game.board, bparms.timerLeft,
|
|
bparms.timerTop, bparms.timerWidth,
|
|
bparms.timerHeight );
|
|
}
|
|
|
|
board_setPos( globals->game.board, bparms.boardLeft,
|
|
bparms.boardTop, XP_FALSE );
|
|
board_setScale( globals->game.board, bparms.boardHScale,
|
|
bparms.boardVScale );
|
|
|
|
board_setScoreboardLoc( globals->game.board, CE_SCORE_LEFT,
|
|
CE_SCORE_TOP, bparms.scoreWidth,
|
|
bparms.scoreHeight, bparms.horiz );
|
|
board_setShowColors( globals->game.board, globals->appPrefs.showColors );
|
|
board_setYOffset( globals->game.board, 0 );
|
|
|
|
board_prefsChanged( globals->game.board, &globals->appPrefs.cp );
|
|
|
|
board_setTrayLoc( globals->game.board, bparms.trayLeft, bparms.trayTop,
|
|
bparms.trayWidth, bparms.trayHeight, CE_DIVIDER_WIDTH );
|
|
|
|
server_prefsChanged( globals->game.server, &globals->appPrefs.cp );
|
|
|
|
return erase;
|
|
} /* cePositionBoard */
|
|
|
|
/* Set the title to be app-name COLON file-name. If there's no colon there
|
|
* now append colon and game name. Else if there is a color replace what
|
|
* follows it with the new name.
|
|
*/
|
|
static void
|
|
ceSetTitleFromName( CEAppGlobals* globals )
|
|
{
|
|
wchar_t widebuf[256];
|
|
wchar_t* colonPos;
|
|
XP_UCHAR* baseStart;
|
|
XP_UCHAR* gameName;
|
|
|
|
XP_U16 len = (XP_U16)SendMessage( globals->hWnd, WM_GETTEXT,
|
|
sizeof(widebuf), (long)widebuf );
|
|
colonPos = wcsstr( widebuf, L":" );
|
|
|
|
if ( colonPos == NULL ) {
|
|
wcscat( widebuf, L":" );
|
|
colonPos = widebuf + len; /* we'll write at the end */
|
|
}
|
|
++colonPos; /* skip the colon */
|
|
|
|
gameName = globals->curGameName;
|
|
baseStart = strrchr( gameName, '\\' );
|
|
++baseStart;
|
|
len = (XP_U16)XP_STRLEN( baseStart );
|
|
|
|
MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, baseStart, len + 1,
|
|
colonPos, len + 1 );
|
|
|
|
/* now get rid of the ".xwg" */
|
|
colonPos = wcsrchr( widebuf, '.' );
|
|
if ( colonPos != NULL ) {
|
|
*colonPos = 0;
|
|
}
|
|
|
|
SendMessage( globals->hWnd, WM_SETTEXT, 0, (long)widebuf );
|
|
} /* ceSetTitleFromName */
|
|
|
|
static void
|
|
ceInitAndStartBoard( CEAppGlobals* globals, XP_Bool newGame,
|
|
const CommsAddrRec* XP_UNUSED_STANDALONE(addr) )
|
|
{
|
|
DictionaryCtxt* dict;
|
|
XP_UCHAR* newDictName = globals->gameInfo.dictName;
|
|
|
|
/* This needs to happen even when !newGame because it's how the dict
|
|
slots in PlayerInfo get loaded. That really ought to happen earlier,
|
|
though. */
|
|
XP_ASSERT( !!globals->game.model );
|
|
dict = model_getDictionary( globals->game.model );
|
|
|
|
if ( !!dict ) {
|
|
const XP_UCHAR* curDictName = dict_getName( dict );
|
|
if ( !!newDictName &&
|
|
( !curDictName || 0 != strcmp( curDictName, newDictName ) ) ) {
|
|
dict_destroy( dict );
|
|
dict = NULL;
|
|
} else {
|
|
replaceStringIfDifferent( globals->mpool,
|
|
&globals->gameInfo.dictName,
|
|
curDictName );
|
|
}
|
|
}
|
|
|
|
if ( !dict ) {
|
|
#ifdef STUBBED_DICT
|
|
dict = make_stubbed_dict( MPPARM_NOCOMMA(globals->mpool) );
|
|
#else
|
|
XP_ASSERT( !!newDictName );
|
|
XP_LOGF( "calling ce_dictionary_make" );
|
|
dict = ce_dictionary_make( globals, newDictName);
|
|
#endif
|
|
XP_ASSERT( !!dict );
|
|
model_setDictionary( globals->game.model, dict );
|
|
}
|
|
|
|
if ( newGame ) {
|
|
XP_U16 newGameID = 0;
|
|
game_reset( MEMPOOL &globals->game, &globals->gameInfo, &globals->util,
|
|
newGameID, &globals->appPrefs.cp, CE_SEND_PROC,
|
|
CE_RESET_PROC globals );
|
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
if ( !!addr ) {
|
|
XP_ASSERT( globals->game.comms != NULL );
|
|
comms_setAddr( globals->game.comms, addr );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
XP_ASSERT( !!globals->game.board );
|
|
#ifdef _WIN32_WCE
|
|
sizeIfFullscreen( globals );
|
|
#endif
|
|
(void)cePositionBoard( globals );
|
|
|
|
board_invalAll( globals->game.board );
|
|
InvalidateRect( globals->hWnd, NULL, TRUE /* erase */ );
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
if ( newGame && globals->gameInfo.serverRole == SERVER_ISCLIENT ) {
|
|
XWStreamCtxt* stream;
|
|
XP_ASSERT( !!globals->game.comms );
|
|
stream = make_generic_stream( globals );
|
|
stream_setOnCloseProc( stream, ce_send_on_close );
|
|
server_initClientConnection( globals->game.server, stream );
|
|
}
|
|
#endif
|
|
|
|
server_do( globals->game.server );
|
|
|
|
globals->isNewGame = FALSE;
|
|
} /* ceInitAndStartBoard */
|
|
|
|
#ifdef DEBUG
|
|
void
|
|
logLastError( const char* comment )
|
|
{
|
|
LPVOID lpMsgBuf;
|
|
DWORD lastErr = GetLastError();
|
|
XP_UCHAR msg[256];
|
|
XP_U16 len;
|
|
XP_U16 lenSoFar;
|
|
|
|
sprintf( msg, "%s (err: %ld): ", comment, lastErr );
|
|
lenSoFar = strlen( msg );
|
|
|
|
FormatMessage(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
NULL,
|
|
lastErr,
|
|
0, // Default language
|
|
(LPTSTR) &lpMsgBuf,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
len = wcslen( lpMsgBuf );
|
|
if ( len >= sizeof(msg)-lenSoFar ) {
|
|
len = sizeof(msg) - lenSoFar - 1;
|
|
}
|
|
WideCharToMultiByte( CP_ACP, 0, lpMsgBuf, len + 1,
|
|
msg + lenSoFar, len + 1, NULL, NULL );
|
|
LocalFree( lpMsgBuf );
|
|
|
|
XP_LOGF( "system error: %s", msg );
|
|
} /* logLastError */
|
|
#endif
|
|
|
|
static void
|
|
ceSavePrefs( CEAppGlobals* globals )
|
|
{
|
|
HANDLE fileH;
|
|
|
|
fileH = CreateFile( PREFSFILENAME, GENERIC_WRITE, 0, NULL,
|
|
OPEN_ALWAYS, FILE_ATTRIBUTE_HIDDEN, NULL );
|
|
if ( fileH != INVALID_HANDLE_VALUE ) {
|
|
XP_U32 nWritten;
|
|
XP_U16 nameLen = 0;
|
|
XP_UCHAR* name = globals->curGameName;
|
|
|
|
if ( name != NULL ) {
|
|
nameLen = (XP_U16)XP_STRLEN( name );
|
|
}
|
|
|
|
SetFilePointer( fileH, 0, 0, FILE_BEGIN );
|
|
/* write prefs, including version num */
|
|
WriteFile( fileH, &globals->appPrefs, sizeof(globals->appPrefs),
|
|
&nWritten, NULL );
|
|
XP_DEBUGF( "sizeof(appPrefs) => %ld", sizeof( globals->appPrefs ) );
|
|
|
|
WriteFile( fileH, &nameLen, sizeof(nameLen), &nWritten, NULL );
|
|
WriteFile( fileH, name, nameLen, &nWritten, NULL );
|
|
|
|
WriteFile( fileH, &globals->flags, sizeof(globals->flags), &nWritten,
|
|
NULL );
|
|
|
|
SetEndOfFile( fileH ); /* truncate anything previously there */
|
|
|
|
CloseHandle( fileH ); /* am I not supposed to do this? PENDING */
|
|
XP_DEBUGF( "ceSavePrefs: prefs file written" );
|
|
} else {
|
|
logLastError( "failed to create prefs file" );
|
|
}
|
|
} /* ceSavePrefs */
|
|
|
|
static XP_Bool
|
|
peekVersion( HANDLE fileH, XP_U16* version )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
XP_U32 nRead;
|
|
success = ReadFile( fileH, version, sizeof(*version), &nRead, NULL );
|
|
if ( success ) {
|
|
SetFilePointer( fileH, -nRead, 0, FILE_CURRENT );
|
|
}
|
|
return success;
|
|
} /* peekVersion */
|
|
|
|
static XP_Bool
|
|
canUpdatePrefs( CEAppGlobals* globals, HANDLE fileH, XP_U16 curVersion,
|
|
CEAppPrefs* prefs )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
LOG_FUNC();
|
|
if ( (curVersion == 0x0002) && (CUR_CE_PREFS_FLAGS == 0x0003) ) {
|
|
CEAppPrefs0002 oldPrefs;
|
|
XP_U32 nRead;
|
|
if ( ReadFile( fileH, &oldPrefs, sizeof(oldPrefs), &nRead, NULL ) ) {
|
|
ceInitPrefs( globals, prefs );
|
|
|
|
XP_MEMCPY( &prefs->cp, &oldPrefs.cp, sizeof(prefs->cp) );
|
|
prefs->showColors = oldPrefs.showColors;
|
|
|
|
XP_MEMCPY( &prefs->colors[0], &oldPrefs.colors[0],
|
|
CE_FOCUS_COLOR*sizeof(prefs->colors[0]));
|
|
XP_ASSERT( CE_USER_COLOR1 - 1 == CE_FOCUS_COLOR );
|
|
XP_MEMCPY( &prefs->colors[CE_USER_COLOR1],
|
|
&oldPrefs.colors[CE_FOCUS_COLOR],
|
|
(CE_NUM_COLORS-CE_USER_COLOR1)
|
|
* sizeof(prefs->colors[0]));
|
|
success = XP_TRUE;
|
|
} else {
|
|
XP_LOGF( "%s: ReadFile bad", __func__ );
|
|
}
|
|
} else {
|
|
XP_LOGF( "%s: can't convert from %d to %d", __func__,
|
|
curVersion, CUR_CE_PREFS_FLAGS );
|
|
}
|
|
LOG_RETURNF( "%d", (int)success );
|
|
return success;
|
|
} /* canUpdatePrefs */
|
|
|
|
static XP_Bool
|
|
ceLoadPrefs( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
HANDLE fileH;
|
|
|
|
fileH = CreateFile( PREFSFILENAME, GENERIC_READ, 0, NULL,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
|
|
if ( fileH != INVALID_HANDLE_VALUE ) {
|
|
XP_U32 fileSize = GetFileSize( fileH, NULL );
|
|
XP_U16 curVersion;
|
|
if ( fileSize >= sizeof(curVersion) && peekVersion( fileH, &curVersion ) ) {
|
|
CEAppPrefs tmpPrefs;
|
|
if ( curVersion == CUR_CE_PREFS_FLAGS ) {
|
|
if ( fileSize >= sizeof( CEAppPrefs ) ) {
|
|
XP_U32 bytesRead;
|
|
if ( ReadFile( fileH, &tmpPrefs, sizeof(tmpPrefs),
|
|
&bytesRead, NULL ) ) {
|
|
|
|
XP_ASSERT( tmpPrefs.versionFlags == CUR_CE_PREFS_FLAGS ) {
|
|
|
|
result = XP_TRUE;
|
|
}
|
|
}
|
|
}
|
|
} else if ( canUpdatePrefs( globals, fileH, curVersion, &tmpPrefs ) ) {
|
|
result = XP_TRUE;
|
|
} else {
|
|
XP_LOGF( "%s: old prefs; cannot read.", __func__ );
|
|
}
|
|
|
|
if ( result ) {
|
|
XP_U16 flags;
|
|
XP_U16 nameLen;
|
|
XP_UCHAR* name;
|
|
XP_U32 nRead;
|
|
|
|
XP_MEMCPY( &globals->appPrefs, &tmpPrefs,
|
|
sizeof(globals->appPrefs) );
|
|
|
|
ReadFile( fileH, &nameLen, sizeof(nameLen), &nRead,
|
|
NULL );
|
|
name = XP_MALLOC( globals->mpool, nameLen + 1 );
|
|
ReadFile( fileH, name, nameLen, &nRead, NULL );
|
|
name[nameLen] = '\0';
|
|
globals->curGameName = name;
|
|
|
|
if ( ReadFile( fileH, &flags, sizeof(flags), &nRead,
|
|
NULL )
|
|
&& nRead == sizeof(flags) ) {
|
|
} else {
|
|
flags = 0;
|
|
}
|
|
globals->flags = flags;
|
|
}
|
|
}
|
|
CloseHandle( fileH );
|
|
} else {
|
|
XP_LOGF( "ceLoadPrefs: prefs file not found" );
|
|
}
|
|
return result; /* none found */
|
|
} /* ceLoadPrefs */
|
|
|
|
static XWStreamCtxt*
|
|
make_generic_stream( CEAppGlobals* globals )
|
|
{
|
|
return mem_stream_make( MPPARM(globals->mpool) globals->vtMgr,
|
|
globals, 0, NULL );
|
|
} /* make_generic_stream */
|
|
|
|
static XWStreamCtxt*
|
|
fileToStream( CEAppGlobals* globals, XP_UCHAR* path )
|
|
{
|
|
XWStreamCtxt* stream = NULL;
|
|
HANDLE fileH;
|
|
wchar_t widebuf[257];
|
|
XP_U16 len;
|
|
|
|
len = (XP_U16)XP_STRLEN( path );
|
|
MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, path, len + 1,
|
|
widebuf, len + 1 );
|
|
|
|
fileH = CreateFile( widebuf, GENERIC_READ, 0, NULL,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
|
|
|
|
if ( fileH != INVALID_HANDLE_VALUE ) {
|
|
XP_U32 len = GetFileSize( fileH, NULL );
|
|
XP_U32 nRead;
|
|
XP_UCHAR* buf = XP_MALLOC( globals->mpool, len );
|
|
|
|
XP_DEBUGF( "fileSize = %ld", len );
|
|
|
|
ReadFile( fileH, buf, len, &nRead, NULL );
|
|
CloseHandle( fileH );
|
|
|
|
stream = make_generic_stream( globals );
|
|
stream_open( stream );
|
|
stream_putBytes( stream, buf, (XP_U16)nRead );
|
|
|
|
XP_FREE( globals->mpool, buf );
|
|
}
|
|
|
|
XP_DEBUGF( "fileToStream => 0x%lx", (unsigned long)stream );
|
|
|
|
return stream;
|
|
} /* fileToStream */
|
|
|
|
static XP_Bool
|
|
ceLoadSavedGame( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
XWStreamCtxt* stream;
|
|
|
|
LOG_FUNC();
|
|
|
|
stream = fileToStream( globals, globals->curGameName );
|
|
success = stream != NULL;
|
|
if ( success ) {
|
|
DictionaryCtxt* dict;
|
|
XP_Bool hasDict;
|
|
|
|
hasDict = stream_getU8( stream );
|
|
if ( hasDict ) {
|
|
#ifdef STUBBED_DICT
|
|
XP_ASSERT(0); /* just don't do this!!!! */
|
|
#else
|
|
XP_UCHAR* name = stringFromStream( globals->mpool, stream );
|
|
dict = ce_dictionary_make( globals, name );
|
|
XP_FREE( globals->mpool, name );
|
|
success = dict != NULL;
|
|
#endif
|
|
} else {
|
|
dict = NULL;
|
|
}
|
|
|
|
if ( success ) {
|
|
XP_DEBUGF( "calling game_makeFromStream" );
|
|
game_makeFromStream( MEMPOOL stream, &globals->game,
|
|
&globals->gameInfo,
|
|
dict, &globals->util, globals->draw,
|
|
&globals->appPrefs.cp, CE_SEND_PROC,
|
|
CE_RESET_PROC globals );
|
|
}
|
|
|
|
stream_destroy( stream );
|
|
}
|
|
|
|
return success;
|
|
} /* ceLoadSavedGame */
|
|
|
|
static void
|
|
colorsFromRsrc( const CEAppGlobals* globals, CEAppPrefs* prefs )
|
|
{
|
|
XP_U16 i;
|
|
HGLOBAL globH;
|
|
HRSRC rsrcH;
|
|
XP_U16* ptr;
|
|
|
|
rsrcH = FindResource( globals->hInst, MAKEINTRESOURCE(ID_COLORS_RES),
|
|
TEXT("CLRS") );
|
|
globH = LoadResource( globals->hInst, rsrcH );
|
|
ptr = (XP_U16*)globH;
|
|
|
|
for ( i = 0; i < CE_NUM_COLORS; ++i ) {
|
|
XP_U8 r = (XP_U8)*ptr++;
|
|
XP_U8 g = (XP_U8)*ptr++;
|
|
XP_U8 b = (XP_U8)*ptr++;
|
|
prefs->colors[i] = RGB( r, g, b );
|
|
}
|
|
|
|
DeleteObject( globH );
|
|
} /* colorsFromRsrc */
|
|
|
|
static void
|
|
ceInitPrefs( CEAppGlobals* globals, CEAppPrefs* prefs )
|
|
{
|
|
prefs->versionFlags = CUR_CE_PREFS_FLAGS;
|
|
prefs->showColors = XP_TRUE;
|
|
prefs->fullScreen = XP_FALSE;
|
|
|
|
prefs->cp.showBoardArrow = XP_TRUE;
|
|
prefs->cp.showRobotScores = XP_FALSE;
|
|
|
|
colorsFromRsrc( globals, prefs );
|
|
|
|
#ifdef DICTS_MOVED_ALERT
|
|
/* The assumption is that if you didn't have prefs already you don't need
|
|
to be told to move dicts. */
|
|
globals->flags = FLAGS_BIT_SHOWN_NEWDICTLOC;
|
|
#endif
|
|
} /* ceInitPrefs */
|
|
|
|
#ifdef DICTS_MOVED_ALERT
|
|
static void
|
|
doDictsMovedAlert( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool hide;
|
|
|
|
XWStreamCtxt* stream = make_generic_stream( globals );
|
|
|
|
stream_putString( stream,
|
|
"Please be aware that starting with this version "
|
|
"Crosswords will not find dictionaries unless they "
|
|
"are located in one of these directories: " );
|
|
ceFormatDictDirs( stream, globals->hInst );
|
|
stream_putString( stream, ". From now on, dictionaries will be "
|
|
"available as .cab files which will put them in the "
|
|
"right place."
|
|
XP_CR
|
|
XP_CR
|
|
"Do you want to disable this warning?" );
|
|
|
|
hide = ceMsgFromStream( globals, stream, L"Warning", XP_TRUE, XP_TRUE );
|
|
|
|
if ( hide ) {
|
|
globals->flags |= FLAGS_BIT_SHOWN_NEWDICTLOC;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef _WIN32_WCE
|
|
static void
|
|
getOSInfo( CEAppGlobals* globals )
|
|
{
|
|
LOG_FUNC();
|
|
TCHAR buf[128];
|
|
|
|
globals->winceVersion = WINCE_UNKNOWN;
|
|
// Check we are running on a Pocket PC
|
|
|
|
if ( SystemParametersInfo( SPI_GETPLATFORMTYPE, sizeof(buf), buf, FALSE ) ) {
|
|
OSVERSIONINFO ver = {0};
|
|
if (GetVersionEx( &ver )) {
|
|
XP_LOGF( "version = %d.%d", ver.dwMajorVersion, ver.dwMinorVersion );
|
|
}
|
|
|
|
if ( lstrcmp( buf, L"PocketPC") == 0 ) {
|
|
// We are on a Pocket PC, so check the OS version,
|
|
// Pocket PC 2003 used WinCE 4.2
|
|
|
|
if ( ver.dwMajorVersion <= 4 ) {
|
|
globals->winceVersion = WINCE_PPC_2003;
|
|
} else if ( ver.dwMajorVersion > 4 ) {
|
|
globals->winceVersion = WINCE_PPC_2005;
|
|
}
|
|
} else {
|
|
XP_LOGW( "unknown OS type", buf );
|
|
}
|
|
}
|
|
LOG_RETURN_VOID();
|
|
}
|
|
#else
|
|
#define getOSInfo( g )
|
|
#endif
|
|
|
|
//
|
|
// FUNCTION: InitInstance(HANDLE, int)
|
|
//
|
|
// PURPOSE: Saves instance handle and creates main window
|
|
//
|
|
// COMMENTS:
|
|
//
|
|
// In this function, we save the instance handle in a global variable and
|
|
// create and display the main program window.
|
|
//
|
|
BOOL
|
|
InitInstance(HINSTANCE hInstance, int nCmdShow)
|
|
{
|
|
HWND hWnd;
|
|
TCHAR szTitle[MAX_LOADSTRING]; // The title bar text
|
|
TCHAR szWindowClass[MAX_LOADSTRING]; // The window class name
|
|
CEAppGlobals* globals;
|
|
BOOL result = TRUE;
|
|
XP_Bool oldGameLoaded;
|
|
XP_Bool prevStateExists;
|
|
XP_Bool newDone = XP_FALSE;
|
|
XP_U16 len;
|
|
MPSLOT;
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
{
|
|
WORD wVersionRequested;
|
|
WSADATA wsaData;
|
|
int err;
|
|
wVersionRequested = MAKEWORD( 2, 2 );
|
|
err = WSAStartup( wVersionRequested, &wsaData );
|
|
if ( err != 0 ) {
|
|
/* Tell the user that we could not find a usable */
|
|
/* WinSock DLL. */
|
|
XP_LOGF( "unable to load winsock" );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
|
|
LoadString(hInstance, IDC_XWORDS4, szWindowClass, MAX_LOADSTRING);
|
|
|
|
//If it is already running, then focus on the window
|
|
hWnd = FindWindow( szWindowClass, szTitle);
|
|
if ( hWnd ) {
|
|
SetForegroundWindow( (HWND)((ULONG) hWnd | 0x00000001) );
|
|
result = FALSE;
|
|
goto exit;
|
|
}
|
|
|
|
(void)CreateDirectory( DEFAULT_DIR_NAME, 0 );
|
|
|
|
#ifdef MEM_DEBUG
|
|
mpool = mpool_make();
|
|
#endif
|
|
|
|
globals = (CEAppGlobals*)XP_MALLOC( mpool, sizeof(*globals) );
|
|
XP_DEBUGF( "" );
|
|
XP_DEBUGF( "globals created: 0x%lx", globals );
|
|
XP_MEMSET( globals, 0, sizeof(*globals) );
|
|
MPASSIGN( globals->mpool, mpool );
|
|
|
|
getOSInfo( globals );
|
|
|
|
#if defined DEBUG && !defined _WIN32_WCE
|
|
globals->dbWidth = g_dbWidth;
|
|
globals->dbHeight = g_dbHeight;
|
|
#endif
|
|
|
|
globals->vtMgr = make_vtablemgr( MPPARM_NOCOMMA(mpool) );
|
|
|
|
len = wcslen( DEFAULT_DIR_NAME );
|
|
len = (len + 1) * sizeof(globals->lastDefaultDir[0]);
|
|
globals->lastDefaultDir = XP_MALLOC( mpool, len );
|
|
XP_MEMCPY( globals->lastDefaultDir, DEFAULT_DIR_NAME, len );
|
|
|
|
globals->hInst = hInstance;
|
|
// Initialize global strings
|
|
MyRegisterClass(hInstance, szWindowClass);
|
|
|
|
hWnd = CreateWindow(szWindowClass, szTitle, WS_VISIBLE,
|
|
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
|
|
CW_USEDEFAULT, NULL, NULL, hInstance, globals);
|
|
globals->hWnd = hWnd;
|
|
|
|
if (!hWnd) {
|
|
result = FALSE;
|
|
goto exit;
|
|
}
|
|
|
|
#ifdef _WIN32_WCE
|
|
if ( globals->hwndCB ) {
|
|
RECT rc, rcmb;
|
|
|
|
GetWindowRect( hWnd, &rc );
|
|
GetWindowRect( globals->hwndCB, &rcmb );
|
|
rc.bottom -= (rcmb.bottom - rcmb.top);
|
|
|
|
MoveWindow(hWnd, rc.left, rc.top, rc.right-rc.left,
|
|
rc.bottom-rc.top, FALSE);
|
|
}
|
|
#endif
|
|
|
|
#ifdef TARGET_OS_WIN32
|
|
srand( time(NULL) );
|
|
#endif
|
|
|
|
ceInitUtilFuncs( globals );
|
|
|
|
gi_initPlayerInfo( MPPARM(mpool) &globals->gameInfo, "Player %d" );
|
|
|
|
/* choose one. If none found it's an error. */
|
|
#ifndef STUBBED_DICT
|
|
result = 1 == ceLocateNDicts( MPPARM(mpool) hInstance, 1, ceSetDictName,
|
|
globals );
|
|
if ( !result ) {
|
|
XWStreamCtxt* stream = make_generic_stream( globals );
|
|
stream_putString( stream, "Please install a Crosswords dictionary "
|
|
"in one of these directories: " );
|
|
ceFormatDictDirs( stream, hInstance );
|
|
stream_putString( stream, ". Download dictionaries from "
|
|
"http://xwords.sf.net." );
|
|
messageBoxStream( globals, stream, L"Dictionary Not Found" );
|
|
stream_destroy( stream );
|
|
result = FALSE;
|
|
goto exit;
|
|
}
|
|
#endif
|
|
|
|
/* here's where we want to behave differently if there's saved state.
|
|
But that's a long ways off. */
|
|
prevStateExists = ceLoadPrefs( globals );
|
|
if ( !prevStateExists ) {
|
|
ceInitPrefs( globals, &globals->appPrefs );
|
|
#ifdef DICTS_MOVED_ALERT
|
|
} else if ( (globals->flags & FLAGS_BIT_SHOWN_NEWDICTLOC) == 0 ) {
|
|
doDictsMovedAlert( globals );
|
|
#endif
|
|
}
|
|
/* must load prefs before creating draw ctxt */
|
|
globals->draw = ce_drawctxt_make( MPPARM(globals->mpool)
|
|
hWnd, globals );
|
|
|
|
oldGameLoaded = prevStateExists && ceLoadSavedGame( globals );
|
|
|
|
if ( !oldGameLoaded ) {
|
|
XP_U16 gameID = 0; /* good enough until I get networking going */
|
|
game_makeNewGame( MPPARM(mpool) &globals->game, &globals->gameInfo,
|
|
&globals->util, globals->draw, gameID,
|
|
&globals->appPrefs.cp,
|
|
CE_SEND_PROC, CE_RESET_PROC globals );
|
|
|
|
newDone = ceDoNewGame( globals ); /* calls ceInitAndStartBoard */
|
|
if ( !newDone ) {
|
|
result = FALSE;
|
|
}
|
|
}
|
|
|
|
ShowWindow(hWnd, nCmdShow);
|
|
UpdateWindow(hWnd);
|
|
#ifdef _WIN32_WCE
|
|
if (globals->hwndCB) {
|
|
CommandBar_Show(globals->hwndCB, TRUE);
|
|
}
|
|
#endif
|
|
if ( result && !newDone ) {
|
|
ceInitAndStartBoard( globals, !oldGameLoaded, NULL );
|
|
}
|
|
|
|
exit:
|
|
return result;
|
|
} /* InitInstance */
|
|
|
|
static XP_Bool
|
|
ceSetDictName( const wchar_t* XP_UNUSED(wPath), XP_U16 XP_UNUSED_DBG(index),
|
|
void* XP_UNUSED(ctxt) )
|
|
{
|
|
/* CEAppGlobals* globals = (CEAppGlobals*)ctxt; */
|
|
/* XP_UCHAR* str; */
|
|
/* XP_UCHAR buf[CE_MAX_PATH_LEN]; */
|
|
XP_ASSERT( index == 0 ); /* we said one only! */
|
|
|
|
/* WideCharToMultiByte( CP_ACP, 0, wPath, -1, */
|
|
/* buf, sizeof(buf)-1, NULL, NULL ); */
|
|
|
|
/* XP_LOGF( "%s: got path \"%s\"", __func__, buf ); */
|
|
/* str = copyString( MPPARM(globals->mpool) buf ); */
|
|
/* XP_LOGF( "%s: got %p", __func__, str ); /\* this is leaking *\/ */
|
|
/* XP_ASSERT( NULL == globals->gameInfo.dictName ); */
|
|
/* globals->gameInfo.dictName = str; */
|
|
return XP_FALSE;
|
|
} /* ceSetDictName */
|
|
|
|
static XP_Bool
|
|
ceHandleHintRequest( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool notDone;
|
|
XP_Bool draw;
|
|
XP_ASSERT( !!globals->game.board );
|
|
|
|
draw = board_requestHint( globals->game.board,
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
globals->askTrayLimits,
|
|
#endif
|
|
¬Done );
|
|
globals->hintPending = notDone;
|
|
if ( draw ) { /* don't turn on if disallowed */
|
|
ceSetLeftSoftkey( globals, ID_MOVE_NEXTHINT );
|
|
}
|
|
return draw;
|
|
} /* ceHandleHintRequest */
|
|
|
|
static XP_Bool
|
|
handleTradeCmd( CEAppGlobals* globals )
|
|
{
|
|
return board_beginTrade( globals->game.board );
|
|
} /* handleTradeCmd */
|
|
|
|
static XP_Bool
|
|
handleJuggleCmd( CEAppGlobals* globals )
|
|
{
|
|
return board_juggleTray( globals->game.board );
|
|
} /* handleJuggleCmd */
|
|
|
|
static XP_Bool
|
|
handleHidetrayCmd( CEAppGlobals* globals )
|
|
{
|
|
XW_TrayVisState curState = board_getTrayVisState( globals->game.board );
|
|
if ( curState == TRAY_REVEALED ) {
|
|
return board_hideTray( globals->game.board );
|
|
} else {
|
|
return board_showTray( globals->game.board );
|
|
}
|
|
} /* handleHidetrayCmd */
|
|
|
|
static XP_Bool
|
|
handleDoneCmd( CEAppGlobals* globals)
|
|
{
|
|
return board_commitTurn( globals->game.board );
|
|
} /* handleDoneCmd */
|
|
|
|
static void
|
|
ceCountsAndValues( CEAppGlobals* globals )
|
|
{
|
|
if ( !!globals->game.server ) {
|
|
XWStreamCtxt* stream = make_generic_stream( globals );
|
|
|
|
server_formatDictCounts( globals->game.server, stream, 3 );
|
|
|
|
(void)ceMsgFromStream( globals, stream, L"Tile Counts and Values",
|
|
XP_FALSE, XP_TRUE );
|
|
}
|
|
} /* ceCountsAndValues */
|
|
|
|
static void
|
|
ceTilesLeft( CEAppGlobals* globals )
|
|
{
|
|
if ( !!globals->game.board ) {
|
|
XWStreamCtxt* stream = make_generic_stream( globals );
|
|
board_formatRemainingTiles( globals->game.board, stream );
|
|
|
|
(void)ceMsgFromStream( globals, stream, L"Remaining tiles",
|
|
XP_FALSE, XP_TRUE );
|
|
}
|
|
} /* ceTilesLeft */
|
|
|
|
static void
|
|
ceDoHistory( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool gameOver = server_getGameIsOver(globals->game.server);
|
|
XWStreamCtxt* stream;
|
|
|
|
stream = make_generic_stream( globals );
|
|
|
|
model_writeGameHistory( globals->game.model, stream,
|
|
globals->game.server, gameOver );
|
|
(void)ceMsgFromStream( globals, stream, L"Game history",
|
|
XP_FALSE, XP_TRUE );
|
|
} /* ceDoHistory */
|
|
|
|
static void
|
|
drawInsidePaint( CEAppGlobals* globals )
|
|
{
|
|
HDC hdc;
|
|
|
|
hdc = GetDC( globals->hWnd );
|
|
if ( !hdc ) {
|
|
logLastError( __func__ );
|
|
} else {
|
|
HDC prevHDC = globals->hdc;
|
|
globals->hdc = hdc;
|
|
|
|
board_draw( globals->game.board );
|
|
|
|
globals->hdc = prevHDC;
|
|
}
|
|
} /* drawInsidePaint */
|
|
|
|
static void
|
|
ceDisplayFinalScores( CEAppGlobals* globals )
|
|
{
|
|
XWStreamCtxt* stream;
|
|
|
|
stream = make_generic_stream( globals );
|
|
server_writeFinalScores( globals->game.server, stream );
|
|
stream_putU8( stream, '\0' );
|
|
|
|
(void)ceMsgFromStream( globals, stream, L"Final scores",
|
|
XP_FALSE, XP_TRUE );
|
|
} /* ceDisplayFinalScores */
|
|
|
|
static XP_Bool
|
|
ceDoNewGame( CEAppGlobals* globals )
|
|
{
|
|
GameInfoState giState;
|
|
CommsAddrRec* addr = NULL;
|
|
XP_Bool changed = XP_FALSE;
|
|
|
|
/* What happens if user cancels below? I'm hosed without a name, no?
|
|
PENDING */
|
|
if ( globals->curGameName != NULL ) {
|
|
XP_FREE( globals->mpool, globals->curGameName );
|
|
globals->curGameName = NULL;
|
|
}
|
|
|
|
XP_MEMSET( &giState, 0, sizeof(giState) );
|
|
giState.globals = globals;
|
|
giState.isNewGame = XP_TRUE;
|
|
|
|
DialogBoxParam( globals->hInst, (LPCTSTR)IDD_GAMEINFO, globals->hWnd,
|
|
(DLGPROC)GameInfo, (long)&giState );
|
|
|
|
if ( !giState.userCancelled
|
|
#ifndef STUBBED_DICT
|
|
&& ( giState.newDictName[0] != '\0' )
|
|
#endif
|
|
) {
|
|
|
|
if ( giState.prefsChanged ) {
|
|
loadCurPrefsFromState( globals, &globals->appPrefs,
|
|
&globals->gameInfo, &giState.prefsPrefs );
|
|
if ( giState.colorsChanged ) {
|
|
updateForColors( globals );
|
|
}
|
|
}
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
if ( giState.addrChanged ) {
|
|
addr = &giState.prefsPrefs.addrRec;
|
|
}
|
|
#endif
|
|
|
|
ceInitAndStartBoard( globals, XP_TRUE, addr );
|
|
ceSetLeftSoftkey( globals, ID_MOVE_TURNDONE );
|
|
changed = XP_TRUE;
|
|
}
|
|
|
|
return changed;
|
|
} /* ceDoNewGame */
|
|
|
|
static void
|
|
ceChooseAndOpen( CEAppGlobals* globals )
|
|
{
|
|
wchar_t path[256];
|
|
OPENFILENAME openFileStruct;
|
|
|
|
XP_MEMSET( &openFileStruct, 0, sizeof(openFileStruct) );
|
|
XP_MEMSET( path, 0, sizeof(path) );
|
|
|
|
openFileStruct.lStructSize = sizeof(openFileStruct);
|
|
openFileStruct.hwndOwner = globals->hWnd;
|
|
openFileStruct.lpstrFilter = L"Crosswords games" L"\0"
|
|
L"*.xwg" L"\0\0";
|
|
openFileStruct.Flags = OFN_FILEMUSTEXIST
|
|
| OFN_HIDEREADONLY
|
|
| OFN_PATHMUSTEXIST;
|
|
|
|
openFileStruct.lpstrFile = path;
|
|
openFileStruct.nMaxFile = VSIZE(path);
|
|
|
|
if ( GetOpenFileName( &openFileStruct ) ) {
|
|
XP_UCHAR* name;
|
|
XP_U16 len;
|
|
|
|
len = wcslen(path);
|
|
name = XP_MALLOC( globals->mpool, len + 1 );
|
|
|
|
WideCharToMultiByte( CP_ACP, 0, path, len + 1,
|
|
name, len + 1, NULL, NULL );
|
|
|
|
if ( globals->curGameName != NULL
|
|
&& 0 == XP_STRCMP( name, globals->curGameName ) ){ /*already open*/
|
|
XP_FREE( globals->mpool, name );
|
|
} else if ( ceSaveCurGame( globals, XP_FALSE )
|
|
|| queryBoxChar( globals, "Do you really want to "
|
|
"overwrite the current game?" ) ) {
|
|
|
|
if ( globals->curGameName != NULL ) {
|
|
XP_FREE( globals->mpool, globals->curGameName );
|
|
}
|
|
|
|
globals->curGameName = name;
|
|
if ( ceLoadSavedGame( globals ) ) {
|
|
ceInitAndStartBoard( globals, XP_FALSE, NULL );
|
|
ceSetTitleFromName( globals );
|
|
} else {
|
|
XP_LOGF( "failed to open chosen game" );
|
|
}
|
|
}
|
|
}
|
|
} /* ceChooseAndOpen */
|
|
|
|
static void
|
|
updateForColors( CEAppGlobals* globals )
|
|
{
|
|
ce_drawctxt_update( globals->draw );
|
|
if ( !!globals->game.board ) {
|
|
board_invalAll( globals->game.board );
|
|
}
|
|
} /* updateForColors */
|
|
|
|
static void
|
|
ceDoPrefsDlg( CEAppGlobals* globals )
|
|
{
|
|
CePrefsDlgState state;
|
|
CePrefsPrefs prefsPrefs;
|
|
|
|
XP_MEMSET( &state, 0, sizeof(state) );
|
|
|
|
loadStateFromCurPrefs( globals, &globals->appPrefs, &globals->gameInfo,
|
|
&prefsPrefs );
|
|
|
|
(void)WrapPrefsDialog( globals->hWnd, globals, &state, &prefsPrefs,
|
|
XP_FALSE );
|
|
|
|
if ( !state.userCancelled ) {
|
|
|
|
loadCurPrefsFromState( globals, &globals->appPrefs, &globals->gameInfo,
|
|
&prefsPrefs );
|
|
|
|
(void)cePositionBoard( globals );
|
|
|
|
if ( state.colorsChanged ) {
|
|
updateForColors( globals );
|
|
}
|
|
/* need to reflect vars set in state into globals, and update/inval
|
|
as appropriate. */
|
|
}
|
|
} /* ceDoPrefsDlg */
|
|
|
|
static void
|
|
ceWriteToFile( XWStreamCtxt* stream, void* closure )
|
|
{
|
|
FileWriteState* fwState = (FileWriteState*)closure;
|
|
#ifdef MEM_DEBUG
|
|
CEAppGlobals* globals = fwState->globals;
|
|
#endif
|
|
XP_UCHAR* path = fwState->path;
|
|
wchar_t widebuf[257];
|
|
XP_U16 len = (XP_U16)XP_STRLEN( path );
|
|
HANDLE fileH;
|
|
|
|
len = (XP_U16)XP_STRLEN( path );
|
|
MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, path, len + 1,
|
|
widebuf, len + 1 );
|
|
|
|
fileH = CreateFile( widebuf, GENERIC_WRITE, 0, NULL,
|
|
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
|
|
|
|
if ( fileH != INVALID_HANDLE_VALUE ) {
|
|
|
|
XP_U16 len = stream_getSize( stream );
|
|
XP_UCHAR* buf = XP_MALLOC( globals->mpool, len );
|
|
XP_U32 nWritten;
|
|
|
|
stream_getBytes( stream, buf, len );
|
|
|
|
WriteFile( fileH, buf, len, &nWritten, NULL );
|
|
SetEndOfFile( fileH );
|
|
CloseHandle( fileH );
|
|
|
|
XP_FREE( globals->mpool, buf );
|
|
}
|
|
} /* ceWriteToFile */
|
|
|
|
static XP_Bool
|
|
isDefaultName( XP_UCHAR* name )
|
|
{
|
|
return 0 == XP_STRCMP( UNSAVEDGAMEFILENAME, name );
|
|
} /* isDefaultName */
|
|
|
|
static void
|
|
makeUniqueName( wchar_t* buf, XP_U16 XP_UNUSED_DBG(bufLen) )
|
|
{
|
|
XP_U16 i;
|
|
DWORD attributes;
|
|
|
|
for ( i = 1; i < 100; ++i ) {
|
|
swprintf( buf, DEFAULT_DIR_NAME L"\\Untitled%d.xwg", i );
|
|
XP_ASSERT( wcslen(buf) < bufLen );
|
|
|
|
attributes = GetFileAttributes( buf );
|
|
if ( attributes == 0xFFFFFFFF ) {
|
|
break;
|
|
}
|
|
}
|
|
/* If we fall out of the loop, the user will be asked to confirm delete
|
|
of Untitled99 or somesuch. That's ok.... */
|
|
} /* makeUniqueName */
|
|
|
|
static XP_Bool
|
|
ceSaveCurGame( CEAppGlobals* globals, XP_Bool autoSave )
|
|
{
|
|
XP_Bool confirmed = XP_FALSE;
|
|
/* If it doesn't yet have a name, get a path at which to save it. User
|
|
has a chance to cancel this. But if there is a name go ahead and save
|
|
using it and don't bother giving cancel chance -- since there's no
|
|
harm in making 'em restart. Not sure how this changes when IR's
|
|
involved. */
|
|
XP_UCHAR* name = globals->curGameName;
|
|
if ( name == NULL || isDefaultName(name) ) {
|
|
wchar_t nameBuf[256];
|
|
XP_UCHAR* newName = NULL;
|
|
|
|
if ( autoSave ) {
|
|
XP_U16 len = XP_STRLEN(UNSAVEDGAMEFILENAME) + 1;
|
|
newName = XP_MALLOC( globals->mpool, len );
|
|
XP_MEMCPY( newName, UNSAVEDGAMEFILENAME, len );
|
|
|
|
confirmed = XP_TRUE;
|
|
} else {
|
|
|
|
OPENFILENAME sfs;
|
|
|
|
XP_MEMSET( &sfs, 0, sizeof(sfs) );
|
|
XP_MEMSET( nameBuf, 0, sizeof(nameBuf) );
|
|
|
|
makeUniqueName( nameBuf, VSIZE(nameBuf) );
|
|
|
|
sfs.lStructSize = sizeof(sfs);
|
|
sfs.Flags = OFN_PATHMUSTEXIST | OFN_OVERWRITEPROMPT;
|
|
sfs.hwndOwner = globals->hWnd;
|
|
sfs.lpstrFile = nameBuf;
|
|
sfs.nMaxFile = VSIZE(nameBuf);
|
|
|
|
sfs.lpstrDefExt = L"xwg";
|
|
|
|
// sfs.lpstrTitle didn't work in earlier PPC OSes, but does now
|
|
sfs.lpstrTitle = L"Save current game as";
|
|
// sfs.lpstrInitialDir: doesn't either
|
|
|
|
confirmed = GetSaveFileName( &sfs );
|
|
|
|
if ( confirmed ) {
|
|
XP_U16 len = wcslen(nameBuf);
|
|
XP_DEBUGF( "len(nameBuf) = %d", len );
|
|
newName = XP_MALLOC( globals->mpool, len + 1 );
|
|
WideCharToMultiByte( CP_ACP, 0, nameBuf, len + 1,
|
|
newName, len + 1, NULL, NULL );
|
|
|
|
/* If user picked from a different directory, remember it
|
|
as the new starting point. */
|
|
nameBuf[sfs.nFileOffset] = 0;
|
|
if ( wcscmp( nameBuf, globals->lastDefaultDir ) != 0 ) {
|
|
XP_FREE( globals->mpool, globals->lastDefaultDir );
|
|
globals->lastDefaultDir =
|
|
XP_MALLOC( globals->mpool,
|
|
(sfs.nFileOffset + 1) * 2 );
|
|
wcscpy( globals->lastDefaultDir, nameBuf );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( confirmed ) {
|
|
XP_ASSERT( !!newName );
|
|
if ( !!globals->curGameName ) {
|
|
XP_FREE( globals->mpool, globals->curGameName );
|
|
}
|
|
globals->curGameName = newName;
|
|
}
|
|
|
|
} else {
|
|
confirmed = XP_TRUE;
|
|
}
|
|
|
|
if ( confirmed ) {
|
|
if ( !!globals->game.server ) {
|
|
XWStreamCtxt* memStream;
|
|
DictionaryCtxt* dict;
|
|
FileWriteState fwState;
|
|
const char* dictName;
|
|
|
|
fwState.path = globals->curGameName;
|
|
fwState.globals = globals;
|
|
board_hideTray( globals->game.board ); /* so won't be visible when
|
|
next opened */
|
|
memStream = mem_stream_make( MEMPOOL globals->vtMgr, &fwState, 0,
|
|
ceWriteToFile );
|
|
stream_open( memStream );
|
|
|
|
/* the dictionary */
|
|
dict = model_getDictionary( globals->game.model );
|
|
#ifdef STUBBED_DICT /* don't check this in!!! */
|
|
dictName = NULL;
|
|
#else
|
|
dictName = !!dict? dict_getName( dict ) : NULL;
|
|
#endif
|
|
stream_putU8( memStream, (XP_U8)!!dictName );
|
|
if ( !!dictName ) {
|
|
stringToStream( memStream, dictName );
|
|
}
|
|
|
|
game_saveToStream( &globals->game, &globals->gameInfo, memStream );
|
|
|
|
stream_destroy( memStream );
|
|
}
|
|
}
|
|
|
|
return confirmed;
|
|
} /* ceSaveCurGame */
|
|
|
|
static XP_Bool
|
|
ceConfirmAndSave( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool confirmed = ceSaveCurGame( globals, XP_TRUE );
|
|
|
|
if ( confirmed ) {
|
|
ceSavePrefs( globals );
|
|
}
|
|
|
|
return confirmed;
|
|
} /* ceConfirmAndSave */
|
|
|
|
static void
|
|
freeGlobals( CEAppGlobals* globals )
|
|
{
|
|
MPSLOT;
|
|
|
|
MPASSIGN( mpool, globals->mpool );
|
|
|
|
draw_destroyCtxt( globals->draw );
|
|
|
|
game_dispose( &globals->game );
|
|
gi_disposePlayerInfo( MPPARM(mpool) &globals->gameInfo );
|
|
|
|
if ( globals->curGameName ) {
|
|
XP_FREE( mpool, globals->curGameName );
|
|
}
|
|
|
|
if ( !!globals->vtMgr ) {
|
|
vtmgr_destroy( MPPARM(mpool) globals->vtMgr );
|
|
}
|
|
if ( !!globals->util.vtable ) {
|
|
XP_FREE( mpool, globals->util.vtable );
|
|
}
|
|
|
|
if ( !!globals->lastDefaultDir ) {
|
|
XP_FREE( mpool, globals->lastDefaultDir );
|
|
}
|
|
|
|
XP_FREE( globals->mpool, globals );
|
|
mpool_destroy( mpool );
|
|
|
|
LOG_RETURN_VOID();
|
|
} /* freeGlobals */
|
|
|
|
#ifdef _WIN32_WCE
|
|
static HWND
|
|
makeCommandBar( HWND hwnd, HINSTANCE hInst )
|
|
{
|
|
SHMENUBARINFO mbi;
|
|
|
|
XP_MEMSET( &mbi, 0, sizeof(mbi) );
|
|
mbi.cbSize = sizeof(mbi);
|
|
mbi.hwndParent = hwnd;
|
|
mbi.nToolBarId = IDM_MAIN_MENUBAR;
|
|
mbi.hInstRes = hInst;
|
|
/* Don't set dwFlags if you want the Wince5 two-button softkey menu. */
|
|
/* mbi.dwFlags = SHCMBF_HMENU; */
|
|
|
|
//mbi.dwFlags = SHCMBF_HIDESIPBUTTON; /* eeh added. Why??? */
|
|
|
|
if (!SHCreateMenuBar(&mbi)) {
|
|
/* will want to use this to change menubar: SHEnableSoftkey? */
|
|
XP_LOGF( "SHCreateMenuBar failed" );
|
|
return NULL;
|
|
}
|
|
|
|
return mbi.hwndMB;
|
|
} /* makeCommandBar */
|
|
#endif
|
|
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
static XP_Bool
|
|
handleScroll( CEAppGlobals* globals, XP_S16 pos, /* only valid for THUMB* */
|
|
XP_S16 code, HWND wnd )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
|
|
if ( wnd == globals->scrollHandle ) {
|
|
XP_S16 newOffset;
|
|
XP_U16 curYOffset = board_getYOffset( globals->game.board );
|
|
|
|
XP_ASSERT( !!globals->game.board );
|
|
|
|
switch ( code ) {
|
|
/* case SB_BOTTOM: // Scrolls to the lower right */
|
|
/* case SB_ENDSCROLL: // Ends scroll */
|
|
|
|
case SB_LINEUP: // Scrolls one line up
|
|
case SB_PAGEUP: //
|
|
newOffset = curYOffset - 1;
|
|
break;
|
|
|
|
case SB_LINEDOWN: // Scrolls one line down
|
|
case SB_PAGEDOWN: // Scrolls one page down
|
|
newOffset = curYOffset + 1;
|
|
break;
|
|
|
|
case SB_THUMBTRACK: /* still dragging; don't redraw */
|
|
case SB_THUMBPOSITION:
|
|
newOffset = pos;
|
|
break;
|
|
default:
|
|
newOffset = -1;
|
|
/* do nothing */
|
|
}
|
|
|
|
if ( newOffset >= 0 && newOffset <= globals->nHiddenRows ) {
|
|
result = curYOffset != newOffset;
|
|
if ( result ) {
|
|
result = board_setYOffset( globals->game.board, newOffset );
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
} /* handleScroll */
|
|
#endif
|
|
|
|
static void
|
|
ceFireTimer( CEAppGlobals* globals, XWTimerReason why )
|
|
{
|
|
XWTimerProc proc;
|
|
void* closure;
|
|
|
|
proc = globals->timerProcs[why];
|
|
if ( !!proc ) {
|
|
globals->timerProcs[why] = NULL;
|
|
closure = globals->timerClosures[why];
|
|
(*proc)( closure, why );
|
|
} else {
|
|
XP_LOGF( "skipped timer; alread fired?" );
|
|
}
|
|
} /* ceFireTimer */
|
|
|
|
/* WM_TIMER messages are low-priority. Hold a key down and key events will
|
|
* crowd it out of the queue so that the app doesn't see it until the key is
|
|
* released. There are more reliable timers, but they seem to require
|
|
* advanced techniques like semaphores. At least one article recommends
|
|
* polling over going to those lengths. This is better that polling. I hope
|
|
* it's enough.
|
|
*/
|
|
static XP_Bool
|
|
checkFireLateKeyTimer( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool drop = XP_FALSE;
|
|
XWTimerReason whys[] = { TIMER_PENDOWN, TIMER_TIMERTICK
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
|
, TIMER_HEARTBEAT
|
|
#endif
|
|
};
|
|
XP_U32 now = GetCurrentTime();
|
|
XP_U16 i;
|
|
|
|
for ( i = 0; i < sizeof(whys)/sizeof(whys[0]); ++i ) {
|
|
XWTimerReason why = whys[i];
|
|
if ( !!globals->timerProcs[why] ) {
|
|
if ( now >= globals->timerWhens[why] ) {
|
|
ceFireTimer( globals, why );
|
|
drop = XP_TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return drop;
|
|
} /* checkFireLateKeyTimer */
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
static XP_Bool
|
|
processPacket( CEAppGlobals* globals, XWStreamCtxt* instream )
|
|
{
|
|
XP_Bool draw = XP_FALSE;
|
|
|
|
XP_ASSERT( globals->game.comms != NULL );
|
|
|
|
if ( comms_checkIncomingStream( globals->game.comms,
|
|
instream, NULL ) ) {
|
|
draw = server_receiveMessage( globals->game.server, instream );
|
|
}
|
|
stream_destroy( instream );
|
|
ce_util_requestTime( &globals->util );
|
|
|
|
return draw;
|
|
} /* processPacket */
|
|
#endif
|
|
|
|
static XP_Bool
|
|
checkPenDown( CEAppGlobals* globals )
|
|
{
|
|
XP_Bool draw = globals->penDown;
|
|
if ( draw ) {
|
|
draw = board_handlePenUp( globals->game.board, 0x7FFF, 0x7FFF );
|
|
globals->penDown = XP_FALSE;
|
|
}
|
|
return draw;
|
|
} /* checkPenDown */
|
|
|
|
#ifdef KEYBOARD_NAV
|
|
|
|
static XP_Bool
|
|
ceCheckHandleFocusKey( CEAppGlobals* globals, WPARAM wParam, LPARAM lParam,
|
|
XP_Bool keyDown, XP_Bool* handledP )
|
|
{
|
|
XP_Bool isRepeat = keyDown && ((HIWORD(lParam) & KF_REPEAT) != 0);
|
|
XP_Key key;
|
|
XP_S16 incr = 0;
|
|
XP_Bool draw = XP_FALSE;
|
|
|
|
switch ( wParam ) {
|
|
case VK_UP:
|
|
key = XP_CURSOR_KEY_UP;
|
|
incr = -1;
|
|
break;
|
|
case VK_RIGHT:
|
|
key = XP_CURSOR_KEY_RIGHT;
|
|
incr = 1;
|
|
break;
|
|
case VK_DOWN:
|
|
key = XP_CURSOR_KEY_DOWN;
|
|
incr = 1;
|
|
break;
|
|
case VK_LEFT:
|
|
key = XP_CURSOR_KEY_LEFT;
|
|
incr = -1;
|
|
break;
|
|
case 0x0d:
|
|
case 0x5d: /* center key on WinMo5 Treo (at least) -- but also ']'*/
|
|
case VK_HOME:
|
|
key = XP_RETURN_KEY;
|
|
if ( isRepeat ) {
|
|
(void)checkFireLateKeyTimer( globals );
|
|
}
|
|
XP_LOGF( "%s: XP_RETURN_KEY", __func__ );
|
|
break;
|
|
|
|
/* Still need to produce these somehow */
|
|
/* XP_CURSOR_KEY_ALTRIGHT, */
|
|
/* XP_CURSOR_KEY_ALTUP, */
|
|
/* XP_CURSOR_KEY_ALTLEFT, */
|
|
/* XP_CURSOR_KEY_ALTDOWN, */
|
|
|
|
default:
|
|
key = XP_KEY_NONE;
|
|
break;
|
|
}
|
|
|
|
if ( key != XP_KEY_NONE ) {
|
|
BoardCtxt* board = globals->game.board;
|
|
|
|
if ( isRepeat ) {
|
|
draw = board_handleKeyRepeat( board, key, handledP );
|
|
} else if ( keyDown ) {
|
|
draw = board_handleKeyDown( board, key, handledP );
|
|
} else {
|
|
draw = board_handleKeyUp( board, key, handledP );
|
|
}
|
|
|
|
if ( !*handledP && incr != 0 && !keyDown ) {
|
|
BoardObjectType order[] = { OBJ_SCORE, OBJ_BOARD, OBJ_TRAY };
|
|
BoardObjectType cur = board_getFocusOwner( board );
|
|
XP_U16 index = 0;
|
|
XP_LOGF( "here" );
|
|
if ( cur != OBJ_NONE ) {
|
|
for ( ; ; ) {
|
|
if ( order[index] == cur ) {
|
|
break;
|
|
}
|
|
++index;
|
|
XP_ASSERT( index < 3 );
|
|
}
|
|
index = (index + 3 + incr) % 3;
|
|
}
|
|
XP_LOGF( "%s: calling board_focusChanged", __func__ );
|
|
draw = board_focusChanged( board, order[index], XP_TRUE );
|
|
}
|
|
}
|
|
return draw;
|
|
} /* ceCheckHandleFocusKey */
|
|
#endif /* KEYBOARD_NAV */
|
|
|
|
#ifdef _WIN32_WCE
|
|
static void
|
|
sizeIfFullscreen( CEAppGlobals* globals )
|
|
{
|
|
RECT rect;
|
|
XP_U16 cbHeight = 0;
|
|
if ( !!globals->hwndCB ) {
|
|
GetWindowRect( globals->hwndCB, &rect );
|
|
cbHeight = rect.bottom - rect.top;
|
|
}
|
|
|
|
/* I'm leaving the SIP/cmdbar in place until I can figure out how to get
|
|
menu events with it hidden -- and also the UI for making sure users
|
|
don't get stuck in fullscreen mode not knowing how to reach menus to
|
|
get out. Later, add SHFS_SHOWSIPBUTTON and SHFS_HIDESIPBUTTON to the
|
|
sets shown and hidden below.*/
|
|
if ( globals->appPrefs.fullScreen ) {
|
|
SHFullScreen( globals->hWnd, SHFS_SHOWTASKBAR | SHFS_SHOWSTARTICON );
|
|
|
|
SystemParametersInfo( SPI_GETWORKAREA, 0, &rect, FALSE );
|
|
} else {
|
|
SHFullScreen( globals->hWnd, SHFS_HIDETASKBAR | SHFS_HIDESTARTICON );
|
|
|
|
SetRect( &rect, 0, 0, GetSystemMetrics(SM_CXSCREEN),
|
|
GetSystemMetrics(SM_CYSCREEN) );
|
|
}
|
|
|
|
rect.bottom -= cbHeight;
|
|
MoveWindow( globals->hWnd, rect.left, rect.top, rect.right - rect.left,
|
|
rect.bottom - rect.top, TRUE );
|
|
} /* sizeIfFullscreen */
|
|
|
|
static void
|
|
ceToggleFullScreen( CEAppGlobals* globals )
|
|
{
|
|
globals->appPrefs.fullScreen = !globals->appPrefs.fullScreen;
|
|
|
|
sizeIfFullscreen( globals );
|
|
|
|
(void)cePositionBoard( globals );
|
|
} /* ceToggleFullScreen */
|
|
#endif
|
|
|
|
LRESULT CALLBACK
|
|
WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT result = 0;
|
|
int wmId;
|
|
RECT rt;
|
|
XP_Bool draw = XP_FALSE;
|
|
XWTimerReason why;
|
|
CEAppGlobals* globals;
|
|
XP_Bool handled;
|
|
XP_Bool callDefault = XP_FALSE;
|
|
|
|
if ( message == WM_CREATE ) {
|
|
globals = ((CREATESTRUCT*)lParam)->lpCreateParams;
|
|
SetWindowLong( hWnd, GWL_USERDATA, (long)globals );
|
|
#ifdef _WIN32_WCE
|
|
globals->hwndCB = makeCommandBar( hWnd, globals->hInst );
|
|
#endif
|
|
|
|
#ifdef _WIN32_WCE
|
|
globals->sai.cbSize = sizeof(globals->sai);
|
|
#endif
|
|
} else {
|
|
/* XP_LOGF( "%s: event=%s (%d)", __func__, messageToStr(message), message ); */
|
|
globals = (CEAppGlobals*)GetWindowLong( hWnd, GWL_USERDATA );
|
|
|
|
switch (message) {
|
|
|
|
#ifdef _WIN32_WCE
|
|
case WM_ACTIVATE:
|
|
// Notify shell of our activate message
|
|
SHHandleWMActivate( hWnd, wParam, lParam, &globals->sai, FALSE );
|
|
break;
|
|
|
|
case WM_SETTINGCHANGE:
|
|
SHHandleWMSettingChange( hWnd, wParam, lParam, &globals->sai );
|
|
if ( !!globals && !!globals->game.model ) {
|
|
cePositionBoard( globals );
|
|
board_invalAll( globals->game.board );
|
|
draw = XP_TRUE;
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
case WM_VSCROLL:
|
|
draw = checkPenDown( globals );
|
|
draw = handleScroll( globals, (short int)HIWORD(wParam),
|
|
(short int)LOWORD(wParam),
|
|
(HWND)lParam ) || draw;
|
|
break;
|
|
#endif
|
|
|
|
case WM_COMMAND:
|
|
(void)checkPenDown( globals );
|
|
wmId = LOWORD(wParam);
|
|
|
|
// Parse the menu selections:
|
|
switch (wmId) {
|
|
case ID_FILE_ABOUT:
|
|
DialogBoxParam(globals->hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd,
|
|
(DLGPROC)ceAbout, 0L );
|
|
break;
|
|
case ID_GAME_GAMEINFO: {
|
|
GameInfoState state;
|
|
|
|
XP_MEMSET( &state, 0, sizeof(state) );
|
|
state.globals = globals;
|
|
|
|
DialogBoxParam(globals->hInst, (LPCTSTR)IDD_GAMEINFO, hWnd,
|
|
(DLGPROC)GameInfo, (long)&state );
|
|
|
|
if ( !state.userCancelled ) {
|
|
if ( state.prefsChanged ) {
|
|
updateForColors( globals );
|
|
}
|
|
draw = server_do( globals->game.server );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case ID_FILE_NEWGAME:
|
|
XP_LOGF( "ID_FILE_NEWGAME" );
|
|
if ( ceSaveCurGame( globals, XP_FALSE )
|
|
|| queryBoxChar( globals, "Do you really want to "
|
|
"overwrite the current game?" ) ) {
|
|
draw = ceDoNewGame( globals );
|
|
}
|
|
break;
|
|
|
|
case ID_FILE_SAVEDGAMES:
|
|
ceChooseAndOpen( globals );
|
|
break;
|
|
|
|
case ID_FILE_PREFERENCES:
|
|
ceDoPrefsDlg( globals );
|
|
break;
|
|
#ifdef _WIN32_WCE
|
|
case ID_FILE_FULLSCREEN:
|
|
ceToggleFullScreen( globals );
|
|
break;
|
|
#endif
|
|
case ID_GAME_FINALSCORES:
|
|
if ( server_getGameIsOver( globals->game.server ) ) {
|
|
ceDisplayFinalScores( globals );
|
|
} else if ( queryBoxChar( globals,
|
|
"Are you sure you want to end "
|
|
"the game now?" ) ) {
|
|
server_endGame( globals->game.server );
|
|
draw = TRUE;
|
|
}
|
|
break;
|
|
|
|
case ID_GAME_TILECOUNTSANDVALUES:
|
|
ceCountsAndValues( globals );
|
|
break;
|
|
|
|
case ID_GAME_TILESLEFT:
|
|
ceTilesLeft( globals );
|
|
break;
|
|
|
|
case ID_GAME_HISTORY:
|
|
ceDoHistory( globals );
|
|
break;
|
|
|
|
case ID_MOVE_TRADE:
|
|
draw = handleTradeCmd( globals );
|
|
break;
|
|
case ID_MOVE_JUGGLE:
|
|
draw = handleJuggleCmd( globals );
|
|
break;
|
|
|
|
case ID_MOVE_HIDETRAY:
|
|
draw = handleHidetrayCmd( globals );
|
|
break;
|
|
case ID_MOVE_TURNDONE:
|
|
draw = handleDoneCmd( globals);
|
|
break;
|
|
|
|
case ID_MOVE_FLIP:
|
|
draw = board_flip( globals->game.board );
|
|
break;
|
|
case ID_MOVE_VALUES:
|
|
draw = board_toggle_showValues( globals->game.board );
|
|
break;
|
|
|
|
case ID_MOVE_HINT:
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
case ID_MOVE_LIMITEDHINT:
|
|
globals->askTrayLimits = wmId == ID_MOVE_LIMITEDHINT;
|
|
#endif
|
|
board_resetEngine( globals->game.board );
|
|
/* fallthru */
|
|
case ID_MOVE_NEXTHINT:
|
|
draw = ceHandleHintRequest( globals );
|
|
break;
|
|
|
|
case ID_FILE_EXIT:
|
|
if ( ceConfirmAndSave( globals ) ) { /* user may cancel... */
|
|
DestroyWindow(hWnd);
|
|
}
|
|
break;
|
|
|
|
case ID_MOVE_UNDOCURRENT:
|
|
draw = board_replaceTiles( globals->game.board );
|
|
break;
|
|
|
|
case ID_MOVE_UNDOLAST:
|
|
draw = server_handleUndo( globals->game.server );
|
|
break;
|
|
|
|
default:
|
|
callDefault = XP_TRUE;
|
|
}
|
|
break;
|
|
case WM_PAINT:
|
|
if ( !!globals && GetUpdateRect( hWnd, &rt, FALSE ) ) {
|
|
XP_Rect rect;
|
|
|
|
if ( !!globals->game.board ) {
|
|
/* When an obscuring window goes away, the update region
|
|
needs to be redrawn. This allows invalidating it. */
|
|
RECTtoXPR( &rect, &rt );
|
|
board_invalRect( globals->game.board, &rect );
|
|
|
|
XP_ASSERT( globals->hWnd == hWnd );
|
|
drawInsidePaint( globals );
|
|
}
|
|
if ( !ValidateRect( hWnd, &rt ) ) {
|
|
logLastError( "WM_PAINT:ValidateRect" );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_LBUTTONDOWN:
|
|
draw = checkPenDown( globals );
|
|
globals->penDown = XP_TRUE;
|
|
draw = board_handlePenDown( globals->game.board, LOWORD(lParam),
|
|
HIWORD(lParam), &handled )
|
|
|| draw;
|
|
break;
|
|
|
|
case WM_MOUSEMOVE:
|
|
if ( globals->penDown ) {
|
|
draw = board_handlePenMove( globals->game.board,
|
|
LOWORD(lParam),
|
|
HIWORD(lParam) );
|
|
}
|
|
break;
|
|
|
|
case WM_LBUTTONUP:
|
|
if ( globals->penDown ) {
|
|
draw = board_handlePenUp( globals->game.board, LOWORD(lParam),
|
|
HIWORD(lParam) );
|
|
globals->penDown = XP_FALSE;
|
|
}
|
|
break;
|
|
|
|
#ifdef KEYBOARD_NAV
|
|
case WM_KEYDOWN:
|
|
case WM_KEYUP:
|
|
draw = ceCheckHandleFocusKey( globals, wParam, lParam,
|
|
message==WM_KEYDOWN, &handled );
|
|
break;
|
|
#endif
|
|
case WM_CHAR:
|
|
if ( wParam == 0x08 ) {
|
|
wParam = XP_CURSOR_KEY_DEL;
|
|
#ifdef KEYBOARD_NAV
|
|
} else if ( wParam == ' ' ) {
|
|
wParam = XP_RAISEFOCUS_KEY;
|
|
#endif
|
|
}
|
|
draw = board_handleKey( globals->game.board, wParam, &handled )
|
|
|| board_handleKey( globals->game.board, wParam - ('a'-'A'),
|
|
&handled );
|
|
break;
|
|
|
|
case WM_TIMER:
|
|
why = (XWTimerReason)wParam;
|
|
if ( why == TIMER_PENDOWN || why == TIMER_TIMERTICK
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
|
|| why == TIMER_HEARTBEAT
|
|
#endif
|
|
) {
|
|
XP_ASSERT( why < NUM_TIMERS_PLUS_ONE );
|
|
|
|
/* Kill since they otherwise repeat, but kill before firing
|
|
as fired proc may set another. */
|
|
(void)KillTimer( hWnd, globals->timerIDs[why] );
|
|
|
|
ceFireTimer( globals, why );
|
|
}
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
#ifdef _WIN32_WCE
|
|
CommandBar_Destroy(globals->hwndCB); /* supposedly not needed */
|
|
#endif
|
|
PostQuitMessage(0);
|
|
freeGlobals( globals );
|
|
break;
|
|
|
|
case XWWM_TIME_RQST:
|
|
draw = server_do( globals->game.server );
|
|
break;
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case XWWM_PACKET_ARRIVED:
|
|
draw = processPacket( globals, (XWStreamCtxt*)lParam );
|
|
break;
|
|
#endif
|
|
|
|
default:
|
|
callDefault = XP_TRUE;
|
|
}
|
|
}
|
|
|
|
if ( callDefault ) {
|
|
result = DefWindowProc(hWnd, message, wParam, lParam );
|
|
} else if ( draw ) {
|
|
/* This is stupid. We can't just say "draw" because windoze clips
|
|
drawing to the inval rect, and the board isn't set up to tell us
|
|
what its inval rect is. So we inval everything, and then when the
|
|
WM_PAINT message comes we inval the whole board because there's a
|
|
huge inval rect. Dumb. Need to figure out how to have the
|
|
methods in cedraw.c set the clip region to encompass the object
|
|
being drawn -- taking board's word for it -- or the intersection
|
|
of that with the actual clip rgn in the case where some window's
|
|
gone away and revealed a large rect board didn't know about. That
|
|
was the source of some trouble on Palm, and CE's so fast this
|
|
works. But it's stupid. */
|
|
RECT r = { 100, 100, 102, 102 };
|
|
InvalidateRect( globals->hWnd, &r, FALSE /* erase */ );
|
|
}
|
|
|
|
return result;
|
|
} /* WndProc */
|
|
|
|
// Mesage handler for the About box.
|
|
LRESULT CALLBACK
|
|
ceAbout(HWND hDlg, UINT message, WPARAM wParam, LPARAM XP_UNUSED(lParam))
|
|
{
|
|
switch (message) {
|
|
case WM_INITDIALOG:
|
|
return TRUE;
|
|
|
|
case WM_COMMAND:
|
|
if ((LOWORD(wParam) == IDOK) || (LOWORD(wParam) == IDCANCEL)) {
|
|
EndDialog(hDlg, LOWORD(wParam));
|
|
return TRUE;
|
|
}
|
|
break;
|
|
}
|
|
return FALSE;
|
|
} /* ceAbout */
|
|
|
|
static XP_Bool
|
|
ceMsgFromStream( CEAppGlobals* globals, XWStreamCtxt* stream,
|
|
wchar_t* title, XP_Bool isQuery, XP_Bool destroy )
|
|
{
|
|
StrBoxInit init;
|
|
|
|
XP_MEMSET( &init, 0, sizeof(init) );
|
|
|
|
init.title = title;
|
|
init.stream = stream;
|
|
init.isQuery = isQuery;
|
|
init.globals = globals;
|
|
|
|
DialogBoxParam( globals->hInst, (LPCTSTR)IDD_STRBOX, globals->hWnd,
|
|
(DLGPROC)StrBox, (long)&init );
|
|
|
|
if ( destroy ) {
|
|
stream_destroy( stream );
|
|
}
|
|
|
|
return init.result == IDOK;
|
|
} /* ceMsgFromStream */
|
|
|
|
static void
|
|
messageBoxChar( CEAppGlobals* globals, XP_UCHAR* str, wchar_t* title )
|
|
{
|
|
wchar_t* widebuf;
|
|
XP_U32 len, wsize;
|
|
|
|
/* first get the length required, then alloc and go */
|
|
len = MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, str, strlen(str),
|
|
NULL, 0 );
|
|
wsize = (len+1) * sizeof( wchar_t );
|
|
|
|
widebuf = XP_MALLOC( globals->mpool, wsize );
|
|
|
|
XP_MEMSET( widebuf, 0, wsize );
|
|
MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, str, strlen(str),
|
|
widebuf, len );
|
|
|
|
MessageBox( globals->hWnd, widebuf, title, MB_OK );
|
|
|
|
XP_FREE( globals->mpool, widebuf );
|
|
} /* messageBoxChar */
|
|
|
|
static XP_UCHAR*
|
|
ceStreamToStrBuf( MPFORMAL XWStreamCtxt* stream )
|
|
{
|
|
XP_U16 len = stream_getSize( stream );
|
|
XP_UCHAR* buf = XP_MALLOC( mpool, len + 1 );
|
|
stream_getBytes( stream, buf, len );
|
|
buf[len] = '\0';
|
|
|
|
return buf;
|
|
} /* ceStreamToStrBuf */
|
|
|
|
static void
|
|
messageBoxStream( CEAppGlobals* globals, XWStreamCtxt* stream, wchar_t* title )
|
|
{
|
|
XP_UCHAR* buf = ceStreamToStrBuf( MPPARM(globals->mpool) stream );
|
|
|
|
messageBoxChar( globals, buf, title );
|
|
|
|
XP_FREE( globals->mpool, buf );
|
|
} /* messageBoxStream */
|
|
|
|
static XP_Bool
|
|
queryBoxChar( CEAppGlobals* globals, XP_UCHAR* msg )
|
|
{
|
|
wchar_t widebuf[128];
|
|
XP_U16 answer;
|
|
|
|
XP_U16 len = MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, msg, strlen(msg),
|
|
widebuf,
|
|
VSIZE(widebuf) );
|
|
widebuf[len] = 0;
|
|
|
|
answer = MessageBox( globals->hWnd, widebuf, L"Question", MB_YESNO );
|
|
return answer == IDOK || answer == IDYES;
|
|
} /* queryBoxChar */
|
|
|
|
static XP_Bool
|
|
ceQueryFromStream( CEAppGlobals* globals, XWStreamCtxt* stream )
|
|
{
|
|
return ceMsgFromStream( globals, stream, L"Question", XP_TRUE,
|
|
XP_FALSE );
|
|
} /* ceQueryFromStream */
|
|
|
|
static void
|
|
RECTtoXPR( XP_Rect* dest, RECT* src )
|
|
{
|
|
dest->top = (short)src->top;
|
|
dest->left = (short)src->left;
|
|
dest->width = (short)(src->right - src->left);
|
|
dest->height = (short)(src->bottom - src->top);
|
|
} /* RECTtoXPR */
|
|
|
|
void
|
|
wince_assert( XP_UCHAR* XP_UNUSED_LOG(s), int XP_UNUSED_LOG(line),
|
|
char* XP_UNUSED_LOG(fileName) )
|
|
{
|
|
XP_WARNF( "ASSERTION FAILED %s: file %s, line %d\n", s, fileName, line );
|
|
} /* wince_assert */
|
|
|
|
#ifdef ENABLE_LOGGING
|
|
static void
|
|
makeTimeStamp( XP_UCHAR* timeStamp, XP_U16 XP_UNUSED_DBG(size) )
|
|
{
|
|
SYSTEMTIME st;
|
|
DWORD tid;
|
|
|
|
tid = GetCurrentThreadId();
|
|
|
|
GetLocalTime( &st );
|
|
sprintf( timeStamp, "<%lx>%d:%.2d:%.2d ", tid, st.wHour, st.wMinute,
|
|
st.wSecond );
|
|
XP_ASSERT( size > strlen(timeStamp) );
|
|
} /* makeTimeStamp */
|
|
|
|
void
|
|
wince_debugf(XP_UCHAR* format, ...)
|
|
{
|
|
#ifdef XWFEATURE_RELAY
|
|
static HANDLE s_logMutex = NULL;
|
|
#endif
|
|
XP_UCHAR buf[256];
|
|
XP_UCHAR timeStamp[32];
|
|
XP_U16 nBytes;
|
|
XP_U32 nWritten;
|
|
HANDLE fileH;
|
|
va_list ap;
|
|
wchar_t* logFileName;
|
|
|
|
va_start( ap, format );
|
|
vsprintf( buf, format, ap );
|
|
va_end(ap);
|
|
|
|
/* Create logfile if necessary and write to it in ascii. If there are
|
|
multiple threads, protect with mutex. */
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
if ( s_logMutex == NULL ) {
|
|
s_logMutex = CreateMutex( NULL, FALSE, NULL );
|
|
}
|
|
if ( WaitForSingleObject( s_logMutex, 200L ) == WAIT_OBJECT_0 ) {
|
|
#endif
|
|
makeTimeStamp(timeStamp, sizeof(timeStamp));
|
|
|
|
#if defined TARGET_OS_WINCE
|
|
logFileName = L"\\My Documents\\Crosswords\\xwDbgLog.txt";
|
|
#elif defined TARGET_OS_WIN32
|
|
logFileName = L"xwDbgLog.txt";
|
|
#endif
|
|
fileH = CreateFile( logFileName,
|
|
GENERIC_WRITE, 0, NULL,
|
|
OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
|
|
|
|
#ifdef _WIN32_WCE_EMULATION
|
|
strcat( buf, "\n" );
|
|
#else
|
|
strcat( buf, XP_CR );
|
|
#endif
|
|
SetFilePointer( fileH, 0, 0, FILE_END );
|
|
#ifdef DEBUG_TS
|
|
nBytes = strlen( timeStamp );
|
|
WriteFile( fileH, timeStamp, nBytes, &nWritten, NULL );
|
|
#endif
|
|
nBytes = strlen( buf );
|
|
WriteFile( fileH, buf, nBytes, &nWritten, NULL );
|
|
CloseHandle( fileH );
|
|
#ifdef XWFEATURE_RELAY
|
|
ReleaseMutex( s_logMutex );
|
|
}
|
|
#endif
|
|
} /* wince_debugf */
|
|
|
|
void
|
|
messageToBuf( UINT message, char* buf, int bufSize )
|
|
{
|
|
char* str = NULL;
|
|
#define STRCASE(s) case s : str = #s; break
|
|
switch( message ) {
|
|
STRCASE(WM_TIMER);
|
|
STRCASE(WM_SETCURSOR);
|
|
STRCASE(WM_NCHITTEST);
|
|
STRCASE(WM_MOUSEMOVE);
|
|
STRCASE(WM_SYSKEYDOWN);
|
|
STRCASE(WM_SYSKEYUP);
|
|
STRCASE(WM_SYSCHAR);
|
|
STRCASE(WM_SYSCOMMAND);
|
|
STRCASE(WM_ENTERMENULOOP);
|
|
STRCASE(WM_INITMENU);
|
|
STRCASE(WM_MENUSELECT);
|
|
STRCASE(WM_COMMAND);
|
|
STRCASE(WM_SETTEXT);
|
|
STRCASE(WM_QUERYNEWPALETTE);
|
|
STRCASE(WM_NCACTIVATE);
|
|
STRCASE(WM_ACTIVATE);
|
|
STRCASE(WM_SHOWWINDOW);
|
|
STRCASE(WM_CTLCOLOREDIT);
|
|
STRCASE(WM_MOUSEACTIVATE);
|
|
STRCASE(WM_CTLCOLORBTN);
|
|
STRCASE(WM_PASTE);
|
|
STRCASE(WM_WINDOWPOSCHANGING);
|
|
STRCASE(WM_SETFOCUS);
|
|
STRCASE(WM_WINDOWPOSCHANGED);
|
|
STRCASE(WM_PAINT);
|
|
STRCASE(WM_VSCROLL);
|
|
STRCASE(WM_LBUTTONDOWN);
|
|
STRCASE(WM_LBUTTONUP);
|
|
STRCASE(WM_INITMENUPOPUP);
|
|
STRCASE(WM_CANCELMODE);
|
|
STRCASE(WM_EXITMENULOOP);
|
|
STRCASE(WM_KILLFOCUS);
|
|
STRCASE(WM_ERASEBKGND);
|
|
default:
|
|
snprintf( buf, bufSize, "%d", message );
|
|
return;
|
|
}
|
|
#undef STRCASE
|
|
if ( !!str ) {
|
|
snprintf( buf, bufSize, "%s", str );
|
|
}
|
|
} /* messageToBuf */
|
|
#endif /* ENABLE_LOGGING */
|
|
|
|
XP_U16
|
|
wince_snprintf( XP_UCHAR* buf, XP_U16 len, const XP_UCHAR* format, ... )
|
|
{
|
|
va_list ap;
|
|
|
|
va_start( ap, format );
|
|
|
|
_vsnprintf( buf, len, format, ap );
|
|
|
|
/* FormatMessage( */
|
|
/* FORMAT_MESSAGE_FROM_STRING, */
|
|
/* format, */
|
|
/* 0, */
|
|
/* 0, // Default language */
|
|
/* (LPTSTR)buf, */
|
|
/* len, &ap ); */
|
|
|
|
va_end(ap);
|
|
|
|
return strlen(buf);
|
|
} /* wince_snprintf */
|
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
static void
|
|
got_data_proc( XP_U8* data, XP_U16 len, void* closure )
|
|
{
|
|
/* Remember that this gets called by the reader thread, not by the one
|
|
running the window loop. */
|
|
CEAppGlobals* globals = (CEAppGlobals*)closure;
|
|
BOOL posted;
|
|
XWStreamCtxt* stream;
|
|
|
|
stream = make_generic_stream( globals );
|
|
stream_putBytes( stream, data, len );
|
|
|
|
posted = PostMessage( globals->hWnd, XWWM_PACKET_ARRIVED,
|
|
0, (DWORD)stream );
|
|
XP_ASSERT( posted );
|
|
} /* got_data_proc */
|
|
#endif
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
static void
|
|
ce_reset_proc( void* closure )
|
|
{
|
|
LOG_FUNC();
|
|
}
|
|
#endif
|
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
static XP_S16
|
|
ce_send_proc( const XP_U8* buf, XP_U16 len, const CommsAddrRec* addr,
|
|
void* closure )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)closure;
|
|
XP_LOGF( "ce_send_proc called" );
|
|
|
|
if ( !globals->socketWrap ) {
|
|
globals->socketWrap = ce_sockwrap_new( MPPARM(globals->mpool)
|
|
addr->conType,
|
|
got_data_proc, globals );
|
|
}
|
|
|
|
return ce_sockwrap_send( globals->socketWrap, buf, len, addr );
|
|
} /* ce_send_proc */
|
|
|
|
static void
|
|
ce_send_on_close( XWStreamCtxt* stream, void* closure )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)closure;
|
|
|
|
XP_ASSERT( !!globals->game.comms );
|
|
comms_send( globals->game.comms, stream );
|
|
}
|
|
#endif
|
|
|
|
static VTableMgr*
|
|
ce_util_getVTManager( XW_UtilCtxt* uc )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
return globals->vtMgr;
|
|
} /* ce_util_getVTManager */
|
|
|
|
static void
|
|
ce_util_userError( XW_UtilCtxt* uc, UtilErrID id )
|
|
{
|
|
XP_UCHAR* message;
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
|
|
switch( id ) {
|
|
case ERR_TILES_NOT_IN_LINE:
|
|
message = "All tiles played must be in a line.";
|
|
break;
|
|
case ERR_NO_EMPTIES_IN_TURN:
|
|
message = "Empty squares cannot separate tiles played.";
|
|
break;
|
|
|
|
case ERR_TWO_TILES_FIRST_MOVE:
|
|
message = "Must play two or more pieces on the first move.";
|
|
break;
|
|
case ERR_TILES_MUST_CONTACT:
|
|
message = "New pieces must contact others already in place (or "
|
|
"the middle square on the first move).";
|
|
break;
|
|
case ERR_NOT_YOUR_TURN:
|
|
message = "You can't do that; it's not your turn!";
|
|
break;
|
|
case ERR_NO_PEEK_ROBOT_TILES:
|
|
message = "No peeking at the robot's tiles!";
|
|
break;
|
|
case ERR_CANT_TRADE_MID_MOVE:
|
|
message = "Remove played tiles before trading.";
|
|
break;
|
|
case ERR_TOO_FEW_TILES_LEFT_TO_TRADE:
|
|
message = "Too few tiles left to trade.";
|
|
break;
|
|
case ERR_CANT_UNDO_TILEASSIGN:
|
|
message = "Tile assignment can't be undone.";
|
|
break;
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case ERR_NO_PEEK_REMOTE_TILES:
|
|
message = "No peeking at remote players' tiles!";
|
|
break;
|
|
case ERR_REG_UNEXPECTED_USER:
|
|
message = "Refused attempt to register unexpected user[s].";
|
|
break;
|
|
case ERR_SERVER_DICT_WINS:
|
|
message = "Conflict between Host and Guest dictionaries; Host wins.";
|
|
XP_WARNF( "GTK may have problems here." );
|
|
break;
|
|
case ERR_REG_SERVER_SANS_REMOTE:
|
|
message = "At least one player must be marked remote for a game "
|
|
"started as Host.";
|
|
break;
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_TIMEOUT:
|
|
message = "The relay timed you out; usually that means "
|
|
"the other players didn't show.";
|
|
break;
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_HEART_YOU:
|
|
message = "You were disconnected from relay because it didn't "
|
|
"hear from you in too long.";
|
|
break;
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_HEART_OTHER:
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_LOST_OTHER:
|
|
message = "The relay has lost contact with a device in this game.";
|
|
break;
|
|
#endif
|
|
|
|
default:
|
|
XP_LOGF( "%s(%d)", __func__, id );
|
|
message = "unknown errorcode ID!!!";
|
|
break;
|
|
}
|
|
|
|
messageBoxChar( globals, message, L"Oops!" );
|
|
|
|
} /* ce_util_userError */
|
|
|
|
static XP_Bool
|
|
ce_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id, XWStreamCtxt* stream )
|
|
{
|
|
char* query = NULL;
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
|
|
switch( id ) {
|
|
case QUERY_COMMIT_TURN:
|
|
return ceQueryFromStream( globals, stream );
|
|
|
|
case QUERY_COMMIT_TRADE:
|
|
query = "Are you sure you want to spend this move trading tiles?";
|
|
return queryBoxChar( globals, query );
|
|
|
|
case QUERY_ROBOT_MOVE:
|
|
return ceMsgFromStream( globals, stream, L"FYI", XP_FALSE,
|
|
XP_FALSE );
|
|
|
|
case QUERY_ROBOT_TRADE:
|
|
messageBoxStream( globals, stream, L"FYI" );
|
|
break;
|
|
|
|
default:
|
|
XP_ASSERT(0);
|
|
}
|
|
|
|
return XP_FALSE;
|
|
} /* ce_util_userQuery */
|
|
|
|
static XWBonusType
|
|
ce_util_getSquareBonus( XW_UtilCtxt* uc, ModelCtxt* XP_UNUSED(model),
|
|
XP_U16 col, XP_U16 row )
|
|
{
|
|
XP_U16 index;
|
|
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
|
|
if ( !globals->bonusInfo ) {
|
|
HRSRC rsrcH;
|
|
HGLOBAL globH;
|
|
|
|
rsrcH = FindResource( globals->hInst, MAKEINTRESOURCE(ID_BONUS_RES),
|
|
TEXT("BONS") );
|
|
if ( !!rsrcH ) {
|
|
globH = LoadResource( globals->hInst, rsrcH );
|
|
|
|
if ( !!globH ) {
|
|
globals->bonusInfo = (XP_U16*)globH;
|
|
/* We don't want to call DeleteObject here, but should when
|
|
the app closes. Or does Wince free up all memory
|
|
associated with a process when it closes? PENDING(eeh) */
|
|
// DeleteObject( globH );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( col > 7 ) col = 14 - col;
|
|
if ( row > 7 ) row = 14 - row;
|
|
index = (row*8) + col;
|
|
|
|
if ( !globals->bonusInfo || (index >= 8*8) ) {
|
|
XP_ASSERT( 0 );
|
|
return (XWBonusType)BONUS_NONE;
|
|
} else {
|
|
/* This is probably a bit slow. Consider caching the resource in
|
|
memory with one bonus value per byte. */
|
|
XP_U16 value = globals->bonusInfo[index/4];
|
|
value >>= ((3 - (index % 4)) * 4);
|
|
return value & 0x0F;
|
|
}
|
|
} /* ce_util_getSquareBonus */
|
|
|
|
static XP_S16
|
|
ce_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* pi,
|
|
XP_U16 playerNum,
|
|
const XP_UCHAR4* texts, XP_U16 nTiles )
|
|
{
|
|
BlankDialogState state;
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
XP_MEMSET( &state, 0, sizeof(state) );
|
|
|
|
state.globals = globals;
|
|
state.texts = texts;
|
|
state.nTiles = nTiles;
|
|
state.playerNum = playerNum;
|
|
state.pi = pi;
|
|
|
|
DialogBoxParam( globals->hInst, (LPCTSTR)IDD_ASKBLANK, globals->hWnd,
|
|
(DLGPROC)BlankDlg, (long)&state );
|
|
return state.result;
|
|
} /* ce_util_userPickTile */
|
|
|
|
static XP_Bool
|
|
ce_util_askPassword( XW_UtilCtxt* uc, const XP_UCHAR* name,
|
|
XP_UCHAR* buf, XP_U16* len )
|
|
{
|
|
PasswdDialogState state;
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
XP_MEMSET( &state, 0, sizeof(state) );
|
|
|
|
state.name = name;
|
|
state.buf = buf;
|
|
state.lenp = len;
|
|
|
|
DialogBoxParam( globals->hInst, (LPCTSTR)IDD_ASKPASS, globals->hWnd,
|
|
(DLGPROC)PasswdDlg, (long)&state );
|
|
|
|
return !state.userCancelled;
|
|
} /* ce_util_askPassword */
|
|
|
|
static void
|
|
ce_util_trayHiddenChange( XW_UtilCtxt* uc, XW_TrayVisState XP_UNUSED(newState),
|
|
XP_U16 nVisibleRows )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
XP_U16 nHiddenRows;
|
|
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
/* If there's a scrollbar, hide/show it. It wants to be
|
|
active/visible only when the tray is NOT hidden */
|
|
|
|
if ( !!globals->scrollHandle ) {
|
|
nHiddenRows = model_numRows( globals->game.model ) - nVisibleRows;
|
|
updateScrollInfo( globals->scrollHandle, nHiddenRows );
|
|
globals->nHiddenRows = nHiddenRows;
|
|
}
|
|
#endif
|
|
|
|
drawInsidePaint( globals );
|
|
} /* ce_util_trayHiddenChange */
|
|
|
|
static void
|
|
ce_util_yOffsetChange( XW_UtilCtxt* uc, XP_U16 XP_UNUSED(oldOffset),
|
|
XP_U16 newOffset )
|
|
{
|
|
#ifdef CEFEATURE_CANSCROLL
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
(void)SetScrollPos( globals->scrollHandle, SB_CTL, newOffset, XP_TRUE );
|
|
#endif
|
|
} /* ce_util_yOffsetChange */
|
|
|
|
static void
|
|
ce_util_turnChanged( XW_UtilCtxt* uc )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
ceSetLeftSoftkey( globals, ID_MOVE_TURNDONE );
|
|
}
|
|
|
|
static void
|
|
ce_util_notifyGameOver( XW_UtilCtxt* uc )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
drawInsidePaint( globals );
|
|
ceDisplayFinalScores( globals );
|
|
|
|
ceSetLeftSoftkey( globals, ID_FILE_NEWGAME );
|
|
} /* ce_util_notifyGameOver */
|
|
|
|
static XP_Bool
|
|
ce_util_hiliteCell( XW_UtilCtxt* XP_UNUSED(uc), XP_U16 XP_UNUSED(col),
|
|
XP_U16 XP_UNUSED(row) )
|
|
{
|
|
return XP_TRUE;
|
|
} /* ce_util_hiliteCell */
|
|
|
|
static XP_Bool
|
|
ce_util_engineProgressCallback( XW_UtilCtxt* XP_UNUSED(uc) )
|
|
{
|
|
return XP_TRUE;
|
|
} /* ce_util_engineProgressCallback */
|
|
|
|
static void
|
|
ce_util_setTimer( XW_UtilCtxt* uc, XWTimerReason why,
|
|
XP_U16 XP_UNUSED_STANDALONE(when), XWTimerProc proc,
|
|
void* closure )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
XP_U32 timerID;
|
|
XP_U32 howLong;
|
|
|
|
XP_ASSERT( why < NUM_TIMERS_PLUS_ONE );
|
|
globals->timerProcs[why] = proc;
|
|
globals->timerClosures[why] = closure;
|
|
|
|
switch ( why ) {
|
|
case TIMER_PENDOWN:
|
|
howLong = 400;
|
|
break;
|
|
case TIMER_TIMERTICK:
|
|
howLong = 1000; /* 1 second */
|
|
break;
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
|
case TIMER_HEARTBEAT:
|
|
howLong = when * 1000;
|
|
break;
|
|
#endif
|
|
default:
|
|
XP_ASSERT(0);
|
|
return;
|
|
}
|
|
|
|
globals->timerWhens[why] = GetCurrentTime() + howLong;
|
|
|
|
timerID = SetTimer( globals->hWnd, why, howLong, NULL);
|
|
|
|
globals->timerIDs[why] = timerID;
|
|
} /* ce_util_setTimer */
|
|
|
|
static void
|
|
ce_util_requestTime( XW_UtilCtxt* uc )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
|
|
PostMessage( globals->hWnd, XWWM_TIME_RQST, 0, 0 );
|
|
} /* palm_util_requestTime */
|
|
|
|
static XP_U32
|
|
ce_util_getCurSeconds( XW_UtilCtxt* XP_UNUSED(uc) )
|
|
{
|
|
/* This function is never called! */
|
|
XP_U32 ticks = GetCurrentTime();
|
|
ticks /= 1000;
|
|
LOG_RETURNF( "%ld", ticks );
|
|
return ticks;
|
|
/* return 0L; */
|
|
} /* ce_util_getCurSeconds */
|
|
|
|
static DictionaryCtxt*
|
|
ce_util_makeEmptyDict( XW_UtilCtxt* uc )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
#ifdef STUBBED_DICT
|
|
return make_stubbed_dict( MPPARM_NOCOMMA(globals->mpool) );
|
|
#else
|
|
return ce_dictionary_make_empty( globals );
|
|
#endif
|
|
} /* ce_util_makeEmptyDict */
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
static XWStreamCtxt*
|
|
ce_util_makeStreamFromAddr( XW_UtilCtxt* uc, XP_PlayerAddr channelNo )
|
|
{
|
|
XWStreamCtxt* stream;
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
|
|
stream = make_generic_stream( globals );
|
|
stream_setOnCloseProc( stream, ce_send_on_close );
|
|
stream_setAddress( stream, channelNo );
|
|
|
|
return stream;
|
|
} /* ce_util_makeStreamFromAddr */
|
|
#endif
|
|
|
|
static const XP_UCHAR*
|
|
ce_util_getUserString( XW_UtilCtxt* XP_UNUSED(uc), XP_U16 stringCode )
|
|
{
|
|
switch( stringCode ) {
|
|
case STRD_REMAINING_TILES_ADD:
|
|
return (XP_UCHAR*)"+ %d [all remaining tiles]";
|
|
case STRD_UNUSED_TILES_SUB:
|
|
return (XP_UCHAR*)"- %d [unused tiles]";
|
|
case STR_BONUS_ALL:
|
|
return (XP_UCHAR*)"Bonus for using all tiles: 50" XP_CR;
|
|
case STRD_TURN_SCORE:
|
|
return (XP_UCHAR*)"Score for turn: %d" XP_CR;
|
|
case STR_COMMIT_CONFIRM:
|
|
return (XP_UCHAR*)"Commit the current move?" XP_CR;
|
|
case STR_LOCAL_NAME:
|
|
return (XP_UCHAR*)"%s";
|
|
case STR_NONLOCAL_NAME:
|
|
return (XP_UCHAR*)"%s (remote)";
|
|
case STRD_TIME_PENALTY_SUB:
|
|
return (XP_UCHAR*)" - %d [time]";
|
|
|
|
case STRD_CUMULATIVE_SCORE:
|
|
return (XP_UCHAR*)"Cumulative score: %d" XP_CR;
|
|
case STRS_MOVE_ACROSS:
|
|
return (XP_UCHAR*)"move (from %s across)" XP_CR;
|
|
case STRS_MOVE_DOWN:
|
|
return (XP_UCHAR*)"move (from %s down)" XP_CR;
|
|
case STRS_TRAY_AT_START:
|
|
return (XP_UCHAR*)"Tray at start: %s" XP_CR;
|
|
|
|
case STRS_NEW_TILES:
|
|
return (XP_UCHAR*)"New tiles: %s" XP_CR;
|
|
case STRSS_TRADED_FOR:
|
|
return (XP_UCHAR*)"Traded %s for %s.";
|
|
case STR_PASS:
|
|
return (XP_UCHAR*)"pass" XP_CR;
|
|
case STR_PHONY_REJECTED:
|
|
return (XP_UCHAR*)"Illegal word in move; turn lost!" XP_CR;
|
|
|
|
case STRD_ROBOT_TRADED:
|
|
return (XP_UCHAR*)"Robot traded tiles %d this turn.";
|
|
case STR_ROBOT_MOVED:
|
|
return (XP_UCHAR*)"The robot made this move:" XP_CR;
|
|
case STR_REMOTE_MOVED:
|
|
return (XP_UCHAR*)"Remote player made this move:" XP_CR;
|
|
|
|
case STR_PASSED:
|
|
return (XP_UCHAR*)"Passed";
|
|
case STRSD_SUMMARYSCORED:
|
|
return (XP_UCHAR*)"%s:%d";
|
|
case STRD_TRADED:
|
|
return (XP_UCHAR*)"Traded %d";
|
|
case STR_LOSTTURN:
|
|
return (XP_UCHAR*)"Lost turn";
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
case STR_LOCALPLAYERS:
|
|
return (XP_UCHAR*)"Locl playrs:";
|
|
#endif
|
|
case STR_TOTALPLAYERS:
|
|
return (XP_UCHAR*)"Player count:";
|
|
|
|
case STRS_VALUES_HEADER:
|
|
return (XP_UCHAR*)"%s counts/values:" XP_CR;
|
|
|
|
default:
|
|
XP_LOGF( "stringCode=%d", stringCode );
|
|
return (XP_UCHAR*)"unknown code";
|
|
}
|
|
} /* ce_util_getUserString */
|
|
|
|
static void
|
|
ce_formatBadWords( BadWordInfo* bwi, XP_UCHAR buf[], XP_U16 bufsiz )
|
|
{
|
|
XP_U16 i;
|
|
|
|
for ( i = 0, buf[0] = '\0'; ; ) {
|
|
XP_UCHAR wordBuf[18];
|
|
sprintf( wordBuf, "\"%s\"", bwi->words[i] );
|
|
XP_ASSERT( strlen(wordBuf) < sizeof(wordBuf)-1 );
|
|
strncat( buf, wordBuf, bufsiz - 1 );
|
|
if ( ++i == bwi->nWords ) {
|
|
break;
|
|
}
|
|
bufsiz -= strlen( wordBuf );
|
|
strncat( buf, ", ", bufsiz - 1 );
|
|
bufsiz -= 2;
|
|
}
|
|
} /* ce_formatBadWords */
|
|
|
|
static XP_Bool
|
|
ce_util_warnIllegalWord( XW_UtilCtxt* uc, BadWordInfo* bwi,
|
|
XP_U16 XP_UNUSED(turn), XP_Bool turnLost )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
XP_UCHAR wordsBuf[256];
|
|
XP_UCHAR msgBuf[256];
|
|
XP_Bool isOk;
|
|
|
|
ce_formatBadWords( bwi, wordsBuf, sizeof(wordsBuf) );
|
|
sprintf( msgBuf, "Word[s] %s not found in dictionary.", wordsBuf );
|
|
|
|
if ( turnLost ) {
|
|
messageBoxChar( globals, msgBuf, L"Illegal word" );
|
|
isOk = XP_TRUE;
|
|
} else {
|
|
strcat( msgBuf, " Use it anyway?" );
|
|
isOk = queryBoxChar( globals, msgBuf );
|
|
}
|
|
|
|
return isOk;
|
|
} /* ce_util_warnIllegalWord */
|
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
|
static void
|
|
ce_util_addrChange( XW_UtilCtxt* XP_UNUSED(uc),
|
|
const CommsAddrRec* XP_UNUSED(oldAddr),
|
|
const CommsAddrRec* XP_UNUSED(newAddr) )
|
|
{
|
|
XP_LOGF( "ce_util_addrChange called; DO SOMETHING." );
|
|
} /* ce_util_addrChange */
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static XP_Bool
|
|
ce_util_getTraySearchLimits( XW_UtilCtxt* uc, XP_U16* min, XP_U16* max )
|
|
{
|
|
CEAppGlobals* globals = (CEAppGlobals*)uc->closure;
|
|
HintLimitsState hls;
|
|
|
|
XP_MEMSET( &hls, 0, sizeof(hls) );
|
|
|
|
hls.globals = globals;
|
|
hls.min = *min;
|
|
hls.max = *max;
|
|
|
|
DialogBoxParam( globals->hInst, (LPCTSTR)IDD_ASKHINTLIMTS, globals->hWnd,
|
|
(DLGPROC)HintLimitsDlg, (long)&hls );
|
|
|
|
if ( !hls.cancelled ) {
|
|
*min = hls.min;
|
|
*max = hls.max;
|
|
}
|
|
|
|
return !hls.cancelled;
|
|
} /* ce_util_getTraySearchLimits */
|
|
#endif
|
|
|
|
#ifdef SHOW_PROGRESS
|
|
static void
|
|
ce_util_engineStarting( XW_UtilCtxt* uc )
|
|
{
|
|
} /* ce_util_engineStarting */
|
|
|
|
static void
|
|
ce_util_engineStopping( XW_UtilCtxt* uc )
|
|
{
|
|
} /* ce_util_engineStopping */
|
|
#endif
|