mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-22 07:28:16 +01:00
484 lines
15 KiB
C
484 lines
15 KiB
C
/* -*- fill-column: 77; compile-command: "make TARGET_OS=wince DEBUG=TRUE" -*- */
|
|
/*
|
|
* Copyright 2004-2008 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 <windowsx.h>
|
|
#include "stdafx.h"
|
|
#include <commdlg.h>
|
|
#include <wchar.h>
|
|
#ifdef _win32_wce
|
|
#include <aygshell.h>
|
|
#endif
|
|
|
|
#include "cemain.h"
|
|
#include "cesvdgms.h"
|
|
#include "ceutil.h"
|
|
#include "cedebug.h"
|
|
#include "debhacks.h"
|
|
|
|
typedef struct CeSaveGameNameState {
|
|
CeDlgHdr dlgHdr;
|
|
wchar_t* buf;
|
|
XP_U16 buflen;
|
|
XP_U16 lableTextId;
|
|
XP_Bool cancelled;
|
|
XP_Bool inited;
|
|
} CeSaveGameNameState;
|
|
|
|
static XP_Bool
|
|
ceFileExists( CEAppGlobals* globals, const wchar_t* name )
|
|
{
|
|
wchar_t buf[CE_MAX_PATH_LEN];
|
|
DWORD attributes;
|
|
XP_U16 len;
|
|
|
|
len = ceGetPath( globals, DEFAULT_DIR_PATH_L, buf, VSIZE(buf) );
|
|
swprintf( &buf[len], L"%s.xwg", name );
|
|
|
|
attributes = GetFileAttributes( buf );
|
|
return attributes != 0xFFFFFFFF;
|
|
}
|
|
|
|
static void
|
|
makeUniqueName( CEAppGlobals* globals, wchar_t* buf,
|
|
XP_U16 XP_UNUSED_DBG(bufLen) )
|
|
{
|
|
XP_U16 ii;
|
|
for ( ii = 1; ii < 100; ++ii ) {
|
|
#ifdef DEBUG
|
|
int len =
|
|
#endif
|
|
swprintf( buf, L"Untitled%d", ii );
|
|
XP_ASSERT( len < bufLen );
|
|
if ( !ceFileExists( globals, buf ) ) {
|
|
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 LRESULT CALLBACK
|
|
SaveNameDlg( HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam )
|
|
{
|
|
CeSaveGameNameState* state;
|
|
XP_U16 wid;
|
|
BOOL result = FALSE;
|
|
|
|
if ( message == WM_INITDIALOG ) {
|
|
SetWindowLongPtr( hDlg, GWL_USERDATA, lParam );
|
|
|
|
state = (CeSaveGameNameState*)lParam;
|
|
state->cancelled = XP_TRUE;
|
|
state->inited = XP_FALSE;
|
|
|
|
wchar_t buf[128];
|
|
LoadString( state->dlgHdr.globals->hInst, state->lableTextId,
|
|
buf, VSIZE(buf) );
|
|
(void)SetDlgItemText( hDlg, IDC_SVGN_SELLAB, buf );
|
|
|
|
ceDlgSetup( &state->dlgHdr, hDlg, DLG_STATE_TRAPBACK );
|
|
|
|
result = TRUE;
|
|
} else {
|
|
state = (CeSaveGameNameState*)GetWindowLongPtr( hDlg, GWL_USERDATA );
|
|
if ( !!state ) {
|
|
CEAppGlobals* globals = state->dlgHdr.globals;
|
|
if ( !state->inited ) {
|
|
state->inited = XP_TRUE;
|
|
(void)SetDlgItemText( hDlg, IDC_SVGN_EDIT, state->buf );
|
|
}
|
|
|
|
if ( ceDoDlgHandle( &state->dlgHdr, message, wParam, lParam) ) {
|
|
result = TRUE;
|
|
} else {
|
|
|
|
switch (message) {
|
|
case WM_COMMAND:
|
|
wid = LOWORD(wParam);
|
|
switch( wid ) {
|
|
case IDOK: {
|
|
wchar_t buf[128];
|
|
XP_U16 len;
|
|
(void)GetDlgItemText( hDlg, IDC_SVGN_EDIT, buf,
|
|
VSIZE(buf) );
|
|
if ( ceFileExists( globals, buf ) ) {
|
|
messageBoxChar( globals,
|
|
"File exists", L"Oops!", MB_OK );
|
|
break;
|
|
}
|
|
len = ceGetPath( globals, DEFAULT_DIR_PATH_L,
|
|
state->buf, state->buflen );
|
|
swprintf( &state->buf[len], L"%s.xwg", buf );
|
|
XP_LOGW( __func__, state->buf );
|
|
/* fallthru */
|
|
state->cancelled = XP_FALSE;
|
|
}
|
|
case IDCANCEL:
|
|
EndDialog(hDlg, wid);
|
|
result = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
} /* SaveNameDlg */
|
|
|
|
XP_Bool
|
|
ceConfirmUniqueName( CEAppGlobals* globals, XP_U16 strId, wchar_t* buf,
|
|
XP_U16 buflen )
|
|
{
|
|
CeSaveGameNameState state;
|
|
|
|
LOG_FUNC();
|
|
|
|
makeUniqueName( globals, buf, buflen );
|
|
|
|
XP_MEMSET( &state, 0, sizeof(state) );
|
|
state.dlgHdr.globals = globals;
|
|
state.buf = buf;
|
|
state.buflen = buflen;
|
|
state.lableTextId = strId;
|
|
(void)DialogBoxParam( globals->hInst, (LPCTSTR)IDD_SAVENAMEDLG,
|
|
globals->hWnd,
|
|
(DLGPROC)SaveNameDlg, (long)&state );
|
|
XP_LOGW( __func__, buf );
|
|
return !state.cancelled;
|
|
} /* ceConfirmUniqueName */
|
|
|
|
typedef struct CeSavedGamesState {
|
|
CeDlgHdr dlgHdr;
|
|
wchar_t* buf;
|
|
XP_U16 buflen;
|
|
XP_S16 sel;
|
|
XP_U16 openGameIndex;
|
|
wchar_t curNameW[128];
|
|
XP_U16 nItems;
|
|
|
|
XP_U16 gameListId;
|
|
XP_Bool opened;
|
|
XP_Bool inited;
|
|
XP_Bool relaunch;
|
|
} CeSavedGamesState;
|
|
|
|
/* Probably belongs as a utility */
|
|
static void
|
|
getComboText( CeSavedGamesState* state, wchar_t* buf, XP_U16* lenp )
|
|
{
|
|
HWND hDlg = state->dlgHdr.hDlg;
|
|
CEAppGlobals* globals = state->dlgHdr.globals;
|
|
XP_U16 id = state->gameListId;
|
|
XP_U16 sel = state->sel;
|
|
XP_U16 len;
|
|
|
|
len = SendDlgItemMessage( hDlg, id, GETLBTEXTLEN(globals), sel, 0L );
|
|
|
|
if ( len < *lenp ) {
|
|
(void)SendDlgItemMessage( hDlg, id, GETLBTEXT(globals), sel,
|
|
(LPARAM)buf );
|
|
} else {
|
|
XP_ASSERT( 0 );
|
|
}
|
|
*lenp = len;
|
|
} /* getComboText */
|
|
|
|
static void
|
|
getFullSelPath( CeSavedGamesState* state, wchar_t* buf, XP_U16 buflen )
|
|
{
|
|
XP_U16 len = ceGetPath( state->dlgHdr.globals,
|
|
DEFAULT_DIR_PATH_L, buf, buflen );
|
|
buflen -= len;
|
|
getComboText( state, &buf[len], &buflen );
|
|
lstrcat( buf, L".xwg" );
|
|
}
|
|
|
|
static void
|
|
setButtons( CeSavedGamesState* state )
|
|
{
|
|
XP_Bool curSelected = state->openGameIndex == state->sel;
|
|
XP_Bool haveItem = state->nItems > 0;
|
|
HWND hDlg = state->dlgHdr.hDlg;
|
|
|
|
ceEnOrDisable( hDlg, IDC_SVGM_OPEN, haveItem && !curSelected );
|
|
ceEnOrDisable( hDlg, IDC_SVGM_DUP, haveItem );
|
|
ceEnOrDisable( hDlg, IDC_SVGM_DEL, haveItem && !curSelected );
|
|
ceEnOrDisable( hDlg, IDC_SVGM_CHANGE, haveItem && !curSelected );
|
|
}
|
|
|
|
static void
|
|
initSavedGamesData( CeSavedGamesState* state )
|
|
{
|
|
HANDLE fileH;
|
|
HWND hDlg = state->dlgHdr.hDlg;
|
|
CEAppGlobals* globals = state->dlgHdr.globals;
|
|
WIN32_FIND_DATA data;
|
|
wchar_t path[CE_MAX_PATH_LEN];
|
|
XP_S16 curSel = -1;
|
|
XP_U16 ii;
|
|
XP_U16 nItems = 0;
|
|
|
|
XP_MEMSET( &data, 0, sizeof(data) );
|
|
ceGetPath( globals, DEFAULT_DIR_PATH_L, path, VSIZE(path) );
|
|
lstrcat( path, L"*.xwg" );
|
|
|
|
fileH = FindFirstFile( path, &data );
|
|
for ( ii = 0; fileH != INVALID_HANDLE_VALUE; ++ii ) {
|
|
XP_U16 len = wcslen( data.cFileName );
|
|
XP_U16 item;
|
|
XP_Bool isCurGame = 0 == wcscmp( state->curNameW, data.cFileName );
|
|
|
|
XP_ASSERT( data.cFileName[len-4] == '.');
|
|
data.cFileName[len-4] = 0;
|
|
|
|
/* Insert in sorted order. This should be fast enough for reasonable
|
|
numbers of saved games. */
|
|
/* PENDING: there's supposed to be a field attribute that make sorted
|
|
order automatic */
|
|
for ( item = 0; item < nItems; ++item ) {
|
|
wchar_t buf[256];
|
|
(void)SendDlgItemMessage( hDlg, state->gameListId,
|
|
GETLBTEXT(globals), item,
|
|
(LPARAM)buf );
|
|
/* Does the current item belong above the one we're inserting? */
|
|
if ( 0 <= wcscmp( buf, data.cFileName ) ) {
|
|
break;
|
|
}
|
|
}
|
|
(void)SendDlgItemMessage( hDlg, state->gameListId,
|
|
INSERTSTRING(globals),
|
|
item, (LPARAM)data.cFileName );
|
|
|
|
/* Remember which entry matches the currently opened game, and adjust
|
|
if it's changed. We may be incrementing an uninitialized curSel,
|
|
but that's ok as isCurGame is guaranteed to be true exactly once
|
|
through. */
|
|
if ( isCurGame ) {
|
|
curSel = item;
|
|
} else if ( curSel >= item ) {
|
|
++curSel; /* since we've moved it up */
|
|
}
|
|
|
|
++nItems;
|
|
|
|
if ( !FindNextFile( fileH, &data ) ) {
|
|
XP_ASSERT( GetLastError() == ERROR_NO_MORE_FILES );
|
|
break;
|
|
}
|
|
}
|
|
|
|
state->nItems = nItems;
|
|
state->openGameIndex = curSel;
|
|
|
|
SendDlgItemMessage( hDlg, state->gameListId,
|
|
SETCURSEL(globals), curSel, 0 );
|
|
state->sel = curSel;
|
|
|
|
setButtons( state );
|
|
|
|
LOG_RETURN_VOID();
|
|
} /* initSavedGamesData */
|
|
|
|
static XP_Bool
|
|
renameSelected( CeSavedGamesState* state )
|
|
{
|
|
wchar_t path[MAX_PATH];
|
|
XP_Bool confirmed = ceConfirmUniqueName( state->dlgHdr.globals, IDS_RENAME,
|
|
path, VSIZE(path) );
|
|
if ( confirmed ) {
|
|
wchar_t curPath[MAX_PATH];
|
|
getFullSelPath( state, curPath, VSIZE(curPath) );
|
|
confirmed = MoveFile( curPath, path );
|
|
}
|
|
return confirmed;
|
|
} /* renameSelected */
|
|
|
|
static XP_Bool
|
|
duplicateSelected( CeSavedGamesState* state )
|
|
{
|
|
wchar_t newPath[MAX_PATH];
|
|
XP_Bool confirmed;
|
|
|
|
confirmed = ceConfirmUniqueName( state->dlgHdr.globals, IDS_DUPENAME,
|
|
newPath, VSIZE(newPath) );
|
|
if ( confirmed ) {
|
|
wchar_t curPath[MAX_PATH];
|
|
getFullSelPath( state, curPath, VSIZE(curPath) );
|
|
confirmed = CopyFile( curPath, newPath, TRUE ); /* TRUE is what??? */
|
|
}
|
|
return confirmed;
|
|
} /* duplicateSelected */
|
|
|
|
static XP_Bool
|
|
deleteSelected( CeSavedGamesState* state )
|
|
{
|
|
/* confirm first!!!! */
|
|
XP_Bool confirmed = queryBoxChar( state->dlgHdr.globals,
|
|
"Are you certain you want to delete the "
|
|
"selected game? This action cannot be "
|
|
"undone.");
|
|
if ( confirmed ) {
|
|
wchar_t pathW[CE_MAX_PATH_LEN];
|
|
XP_U16 len = ceGetPath( state->dlgHdr.globals,
|
|
DEFAULT_DIR_PATH_L, pathW, VSIZE(pathW) );
|
|
XP_U16 remLen = VSIZE(pathW) - len;
|
|
getComboText( state, &pathW[len], &remLen );
|
|
wcscat( pathW, L".xwg" );
|
|
confirmed = DeleteFile( pathW );
|
|
if ( confirmed ) {
|
|
state->sel = -1;
|
|
}
|
|
}
|
|
return confirmed;
|
|
} /* deleteSelected */
|
|
|
|
static XP_Bool
|
|
tryGameChanged( CeSavedGamesState* state )
|
|
{
|
|
XP_S16 sel = SendDlgItemMessage( state->dlgHdr.hDlg, state->gameListId,
|
|
GETCURSEL(state->dlgHdr.globals), 0, 0L);
|
|
XP_Bool changing = sel >= 0 && state->sel != sel;
|
|
if ( changing ) {
|
|
state->sel = sel;
|
|
setButtons( state );
|
|
}
|
|
return changing;
|
|
} /* tryGameChanged */
|
|
|
|
static LRESULT CALLBACK
|
|
SavedGamesDlg( HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam )
|
|
{
|
|
CeSavedGamesState* state;
|
|
BOOL result = FALSE;
|
|
|
|
if ( message == WM_INITDIALOG ) {
|
|
SetWindowLongPtr( hDlg, GWL_USERDATA, lParam );
|
|
|
|
state = (CeSavedGamesState*)lParam;
|
|
state->inited = XP_FALSE;
|
|
state->gameListId = LB_IF_PPC(state->dlgHdr.globals,IDC_SVGM_GAMELIST);
|
|
|
|
ceDlgSetup( &state->dlgHdr, hDlg, DLG_STATE_DONEONLY );
|
|
ceDlgComboShowHide( &state->dlgHdr, IDC_SVGM_GAMELIST );
|
|
|
|
result = TRUE;
|
|
} else {
|
|
state = (CeSavedGamesState*)GetWindowLongPtr( hDlg, GWL_USERDATA );
|
|
if ( !!state ) {
|
|
|
|
if ( !state->inited ) {
|
|
state->inited = XP_TRUE;
|
|
initSavedGamesData( state );
|
|
}
|
|
|
|
if ( ceDoDlgHandle( &state->dlgHdr, message, wParam, lParam) ) {
|
|
result = TRUE;
|
|
} else if ( WM_NOTIFY == message ) {
|
|
result = tryGameChanged( state );
|
|
} else if ( message == WM_COMMAND ) {
|
|
XP_U16 wid = LOWORD(wParam);
|
|
|
|
if ( CBN_SELCHANGE == HIWORD(wParam) ) {
|
|
if (state->gameListId == wid ) {
|
|
result = tryGameChanged( state );
|
|
}
|
|
} else if ( BN_CLICKED == HIWORD(wParam) ) {
|
|
switch( wid ) {
|
|
case IDC_SVGM_DUP:
|
|
state->relaunch = duplicateSelected( state );
|
|
break;
|
|
case IDC_SVGM_CHANGE:
|
|
state->relaunch = renameSelected( state );
|
|
break;
|
|
case IDC_SVGM_DEL:
|
|
state->relaunch = deleteSelected( state );
|
|
break;
|
|
|
|
case IDC_SVGM_OPEN: {
|
|
wchar_t buf[128];
|
|
XP_U16 len = VSIZE(buf);
|
|
getComboText( state, buf, &len );
|
|
len = ceGetPath( state->dlgHdr.globals, DEFAULT_DIR_PATH_L,
|
|
state->buf, state->buflen );
|
|
swprintf( &state->buf[len], L"%s.xwg", buf );
|
|
XP_LOGW( "returning", state->buf );
|
|
state->opened = XP_TRUE;
|
|
}
|
|
/* fallthrough */
|
|
case IDOK: /* Done button */
|
|
EndDialog(hDlg, wid);
|
|
result = TRUE;
|
|
break;
|
|
}
|
|
|
|
if ( state->relaunch ) {
|
|
EndDialog( hDlg, wid );
|
|
result = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
} /* SavedGamesDlg */
|
|
|
|
XP_Bool
|
|
ceSavedGamesDlg( CEAppGlobals* globals, const XP_UCHAR* curPath,
|
|
wchar_t* buf, XP_U16 buflen )
|
|
{
|
|
CeSavedGamesState state;
|
|
|
|
LOG_FUNC();
|
|
|
|
XP_MEMSET( &state, 0, sizeof(state) ); /* sets cancelled */
|
|
state.dlgHdr.globals = globals;
|
|
state.buf = buf;
|
|
state.buflen = buflen;
|
|
state.sel = -1;
|
|
|
|
if ( !!curPath ) {
|
|
wchar_t shortName[128];
|
|
XP_U16 len;
|
|
XP_U16 dirLen;
|
|
|
|
dirLen = ceGetPath( globals, DEFAULT_DIR_PATH_L, shortName,
|
|
VSIZE(shortName) );
|
|
|
|
len = (XP_U16)XP_STRLEN( curPath );
|
|
MultiByteToWideChar( CP_ACP, MB_PRECOMPOSED, curPath, len + 1,
|
|
shortName, len + 1 );
|
|
lstrcpy( state.curNameW, shortName+dirLen );
|
|
}
|
|
|
|
for ( ; ; ) {
|
|
(void)DialogBoxParam( globals->hInst, (LPCTSTR)IDD_SAVEDGAMESDLG,
|
|
globals->hWnd,
|
|
(DLGPROC)SavedGamesDlg, (long)&state );
|
|
|
|
if ( !state.relaunch ) {
|
|
break;
|
|
}
|
|
state.relaunch = XP_FALSE;
|
|
}
|
|
XP_LOGW( __func__, buf );
|
|
|
|
return state.opened;
|
|
} /* ceSavedGamesDlg */
|