source-engine/utils/xbox/vxconsole/exclude_paths.cpp

1135 lines
27 KiB
C++
Raw Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// EXCLUDE_PATHS.CPP
//
// DVD Exclude paths.
//=====================================================================================//
#include "vxconsole.h"
#define UM_CHECKSTATECHANGE (WM_USER + 100)
#define UM_FIRSTTIMEPOPULATE (WM_USER + 101)
#define MAX_TREE_DEPTH 32
#define ROOT_NAME "Xbox:\\"
#define EXCLUDEPATHS_FILE "xbox_exclude_paths.txt"
// known shipping 360 gamedirs
struct GameName_t
{
const char *pName;
bool bIsModPath;
};
struct GamePath_t
{
CUtlString pathName;
bool bIsModPath;
};
GameName_t g_GameNames[] =
{
{ "bin", false },
{ "platform", false },
{ "tf", true },
{ "portal", true },
{ "hl2", true },
{ "episodic", true },
{ "ep2", true }
};
CUtlVector< CUtlString > g_ExcludePaths;
BOOL g_bLinkGameDirs;
inline bool PathLessThan( GamePath_t const &lhs, GamePath_t const &rhs )
{
if ( lhs.bIsModPath != rhs.bIsModPath )
{
// sort mod paths to the bottom
return ( lhs.bIsModPath == false );
}
return ( stricmp( lhs.pathName.String(), rhs.pathName.String() ) < 0 );
}
CUtlRBTree< GamePath_t > g_PathTable( 0, 0, PathLessThan );
//-----------------------------------------------------------------------------
// Add string to TreeView at specified level. Assumes an inorder insert.
// A node (at specified depth) must be inserted before it's children.
//-----------------------------------------------------------------------------
static HTREEITEM AddItemToTree( HWND hWndTree, LPSTR lpszItem, int nLevel, bool bIsModPath )
{
TVITEM tvi = { 0 };
TVINSERTSTRUCT tvins = { 0 };
static HTREEITEM s_hPrevItems[MAX_TREE_DEPTH];
if ( nLevel < 0 || nLevel >= MAX_TREE_DEPTH )
{
Assert( 0 );
return NULL;
}
HTREEITEM hParent;
HTREEITEM hPrev;
if ( !nLevel )
{
// one root item only, reset
for ( int i = 0; i < ARRAYSIZE( s_hPrevItems ); i++ )
{
s_hPrevItems[i] = NULL;
}
hParent = TVI_ROOT;
hPrev = (HTREEITEM)TVI_FIRST;
}
else
{
hParent = s_hPrevItems[nLevel-1];
hPrev = s_hPrevItems[nLevel];
if ( !hParent )
{
// parent should have been setup
Assert( 0 );
return NULL;
}
if ( !hPrev )
{
hPrev = TVI_FIRST;
}
}
tvi.mask = TVIF_TEXT | TVIF_PARAM;
tvi.pszText = lpszItem;
tvi.cchTextMax = sizeof(tvi.pszText)/sizeof(tvi.pszText[0]);
tvi.lParam = (LPARAM)MAKELONG( nLevel, bIsModPath );
tvins.item = tvi;
tvins.hParent = hParent;
tvins.hInsertAfter = hPrev;
// Add the item to the tree-view control.
hPrev = TreeView_InsertItem( hWndTree, &tvins );
s_hPrevItems[nLevel] = hPrev;
if ( nLevel > 0 )
{
// set a back link to its parent
tvi.mask = TVIF_HANDLE;
tvi.hItem = TreeView_GetParent( hWndTree, hPrev );
TreeView_SetItem( hWndTree, &tvi );
}
return hPrev;
}
//-----------------------------------------------------------------------------
// Purpose: Get list of files from current path that match pattern
//-----------------------------------------------------------------------------
static int GetFileList( const char* pDirPath, const char* pPattern, CUtlVector< CUtlString > &fileList )
{
char sourcePath[MAX_PATH];
char fullPath[MAX_PATH];
bool bFindDirs;
fileList.Purge();
strcpy( sourcePath, pDirPath );
int len = (int)strlen( sourcePath );
if ( !len )
{
strcpy( sourcePath, ".\\" );
}
else if ( sourcePath[len-1] != '\\' )
{
sourcePath[len] = '\\';
sourcePath[len+1] = '\0';
}
strcpy( fullPath, sourcePath );
if ( pPattern[0] == '\\' && pPattern[1] == '\0' )
{
// find directories only
bFindDirs = true;
strcat( fullPath, "*" );
}
else
{
// find files, use provided pattern
bFindDirs = false;
strcat( fullPath, pPattern );
}
struct _finddata_t findData;
intptr_t h = _findfirst( fullPath, &findData );
if ( h == -1 )
{
return 0;
}
do
{
// dos attribute complexities i.e. _A_NORMAL is 0
if ( bFindDirs )
{
// skip non dirs
if ( !( findData.attrib & _A_SUBDIR ) )
continue;
}
else
{
// skip dirs
if ( findData.attrib & _A_SUBDIR )
continue;
}
if ( !stricmp( findData.name, "." ) )
continue;
if ( !stricmp( findData.name, ".." ) )
continue;
char fileName[MAX_PATH];
strcpy( fileName, sourcePath );
strcat( fileName, findData.name );
int j = fileList.AddToTail();
fileList[j].Set( fileName );
}
while ( !_findnext( h, &findData ) );
_findclose( h );
return fileList.Count();
}
//-----------------------------------------------------------------------------
// Purpose: Recursively determine directory tree
//-----------------------------------------------------------------------------
static void RecurseFileTree_r( const char *pBasePath, const char *pDirPath, int depth, CUtlVector< CUtlString > &dirList, bool bIsModPath )
{
if ( depth >= 2 )
{
// too much unecessary detail
return;
}
// ignore path roots, only interested in subdirs
const char *pSubName = pDirPath + strlen( pBasePath );
if ( pSubName[0] )
{
GamePath_t gamePath;
gamePath.pathName = pSubName;
gamePath.bIsModPath = bIsModPath;
int iIndex = g_PathTable.Find( gamePath );
if ( iIndex == g_PathTable.InvalidIndex() )
{
g_PathTable.Insert( gamePath );
}
}
// recurse from source directory, get directories only
CUtlVector< CUtlString > fileList;
int dirCount = GetFileList( pDirPath, "\\", fileList );
if ( !dirCount )
{
// add directory name to search tree
int j = dirList.AddToTail();
dirList[j].Set( pDirPath );
return;
}
for ( int i=0; i<dirCount; i++ )
{
// form new path name, recurse into
RecurseFileTree_r( pBasePath, fileList[i].String(), depth+1, dirList, bIsModPath );
}
int j = dirList.AddToTail();
dirList[j].Set( pDirPath );
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_SetCheckState_r
//
// Propogate the check state to all children
//-----------------------------------------------------------------------------
void ExcludePathsDlg_SetCheckState_r( HWND hWndTree, HTREEITEM hTree, int depth, int checkState )
{
if ( !hTree )
{
return;
}
TreeView_SetCheckState( hWndTree, hTree, ( checkState == 1 ) );
TVITEM tvi = { 0 };
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN;
tvi.hItem = hTree;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
ExcludePathsDlg_SetCheckState_r( hWndTree, hChild, depth+1, checkState );
}
}
}
else
{
return;
}
if ( !depth )
{
// only iterate siblings of the parent's child
return;
}
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
return;
}
TreeView_SetCheckState( hWndTree, hSibling, ( checkState == 1 ) );
tvi.hItem = hSibling;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
ExcludePathsDlg_SetCheckState_r( hWndTree, hChild, depth+1, checkState );
}
}
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_SetCheckStateLinked_r
//
// Propogate the check state to all "mod path" children that match the specified name.
// A NULL name matches all.
//-----------------------------------------------------------------------------
void ExcludePathsDlg_SetCheckStateLinked_r( HWND hWndTree, HTREEITEM hTree, int depth, int checkState, const char *pName )
{
if ( !hTree )
{
return;
}
char szNodeName[MAX_PATH];
TVITEM tvi = { 0 };
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN | TVIF_TEXT | TVIF_PARAM;
tvi.hItem = hTree;
tvi.pszText = szNodeName;
tvi.cchTextMax = sizeof( szNodeName );
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
bool bIsModPath = HIWORD( tvi.lParam ) != 0;
if ( bIsModPath && ( !pName || !V_stricmp( szNodeName, pName ) ) )
{
TreeView_SetCheckState( hWndTree, hTree, ( checkState == 1 ) );
}
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
ExcludePathsDlg_SetCheckStateLinked_r( hWndTree, hChild, depth+1, checkState, pName );
}
}
}
else
{
return;
}
if ( !depth )
{
// only iterate siblings of the parent's child
return;
}
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
return;
}
tvi.hItem = hSibling;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
bool bIsModPath = HIWORD( tvi.lParam ) != 0;
if ( bIsModPath && ( !pName || !V_stricmp( szNodeName, pName ) ) )
{
TreeView_SetCheckState( hWndTree, hSibling, ( checkState == 1 ) );
}
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
ExcludePathsDlg_SetCheckStateLinked_r( hWndTree, hChild, depth+1, checkState, pName );
}
}
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_GetCheckState_r
//
// Caller invokes at node. Returns with state set.
// State 0: unchecked, 1: checked, -1:unknown.
// Returns true if All children match, false otherwise.
//-----------------------------------------------------------------------------
bool ExcludePathsDlg_GetCheckState_r( HWND hWndTree, HTREEITEM hTree, int depth, int *pCheckState )
{
int checkState;
checkState = TreeView_GetCheckState( hWndTree, hTree );
if ( *pCheckState == -1 )
{
*pCheckState = checkState;
}
else if ( *pCheckState != checkState )
{
// disparate state, no need to recurse
return false;
}
TVITEM tvi = { 0 };
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN;
tvi.hItem = hTree;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
if ( !ExcludePathsDlg_GetCheckState_r( hWndTree, hChild, depth+1, pCheckState ) )
{
return false;
}
}
}
}
else
{
return false;
}
if ( !depth )
{
// only iterate siblings of the parent's child
return true;
}
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
break;
}
checkState = TreeView_GetCheckState( hWndTree, hSibling );
if ( *pCheckState != checkState )
{
// disparate state, no need to recurse
return false;
}
tvi.hItem = hSibling;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
if ( !ExcludePathsDlg_GetCheckState_r( hWndTree, hChild, depth+1, pCheckState ) )
{
return false;
}
}
}
}
}
return true;
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Expand_r
//
// MaxDepth >= 0, Expand/Collapse up to specified depth.
//-----------------------------------------------------------------------------
void ExcludePathsDlg_Expand_r( HWND hWndTree, HTREEITEM hTree, int depth, int maxDepth, bool bExpand )
{
if ( !hTree )
{
return;
}
if ( maxDepth >= 0 && depth >= maxDepth )
{
return;
}
int flags;
if ( bExpand )
flags = TVE_EXPAND;
else
flags = TVE_COLLAPSE;
TVITEM tvi = { 0 };
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN;
tvi.hItem = hTree;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
TreeView_Expand( hWndTree, hTree, flags );
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
ExcludePathsDlg_Expand_r( hWndTree, hChild, depth+1, maxDepth, bExpand );
}
}
}
else
{
return;
}
if ( !depth )
{
// only iterate siblings of the parent's child
return;
}
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
return;
}
tvi.hItem = hSibling;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
if ( tvi.cChildren )
{
TreeView_Expand( hWndTree, hSibling, flags );
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
ExcludePathsDlg_Expand_r( hWndTree, hChild, depth+1, maxDepth, bExpand );
}
}
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_ExpandToShowState_r
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_ExpandToShowState_r( HWND hWndTree, HTREEITEM hTree, int depth )
{
if ( !hTree )
{
return;
}
TVITEM tvi = { 0 };
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN;
tvi.hItem = hTree;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
int checkState = -1;
bool bStateIsSame = ExcludePathsDlg_GetCheckState_r( hWndTree, hTree, 0, &checkState );
if ( tvi.cChildren && !bStateIsSame )
{
TreeView_Expand( hWndTree, hTree, TVE_EXPAND );
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
ExcludePathsDlg_ExpandToShowState_r( hWndTree, hChild, depth+1 );
}
}
}
else
{
return;
}
if ( !depth )
{
// only iterate siblings of the parent's child
return;
}
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
return;
}
tvi.hItem = hSibling;
if ( TreeView_GetItem( hWndTree, &tvi ) )
{
int checkState = -1;
bool bStateIsSame = ExcludePathsDlg_GetCheckState_r( hWndTree, hSibling, 0, &checkState );
if ( tvi.cChildren && !bStateIsSame )
{
TreeView_Expand( hWndTree, hSibling, TVE_EXPAND );
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
ExcludePathsDlg_ExpandToShowState_r( hWndTree, hChild, depth+1 );
}
}
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Find_r
//
//-----------------------------------------------------------------------------
HTREEITEM ExcludePathsDlg_Find_r( HWND hWndTree, HTREEITEM hTree, int depth, const char *pBasePath, const char *pFindPath )
{
TVITEM tvi = { 0 };
char szName[MAX_PATH];
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN | TVIF_TEXT;
tvi.hItem = hTree;
tvi.pszText = szName;
tvi.cchTextMax = sizeof( szName );
if ( !TreeView_GetItem( hWndTree, &tvi ) )
{
return NULL;
}
char szPath[MAX_PATH];
V_ComposeFileName( pBasePath, szName, szPath, sizeof( szPath ) );
if ( !stricmp( szPath, pFindPath ) )
{
return hTree;
}
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
HTREEITEM hFindTree = ExcludePathsDlg_Find_r( hWndTree, hChild, depth+1, szPath, pFindPath );
if ( hFindTree )
{
return hFindTree;
}
}
}
if ( !depth )
{
// only iterate siblings of the parent's child
return NULL;
}
// iterate siblings
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
break;
}
tvi.hItem = hSibling;
if ( !TreeView_GetItem( hWndTree, &tvi ) )
{
break;
}
V_ComposeFileName( pBasePath, szName, szPath, sizeof( szPath ) );
if ( !stricmp( szPath, pFindPath ) )
{
return hSibling;
}
if ( tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
HTREEITEM hFindTree = ExcludePathsDlg_Find_r( hWndTree, hChild, depth+1, szPath, pFindPath );
if ( hFindTree )
{
return hFindTree;
}
}
}
}
return NULL;
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_BuildExcludeList_r
//
// An exclude path represents the path head, and thus does not need to iterate its children
// if they are deemed to the same exclusion state.
//-----------------------------------------------------------------------------
void ExcludePathsDlg_BuildExcludeList_r( HWND hWndTree, HTREEITEM hTree, int depth, const char *pPath )
{
int checkState = -1;
bool bStateIsSame = ExcludePathsDlg_GetCheckState_r( hWndTree, hTree, 0, &checkState );
if ( checkState == -1 )
{
return;
}
TVITEM tvi = { 0 };
char szName[MAX_PATH];
tvi.mask = TVIF_HANDLE | TVIF_CHILDREN | TVIF_TEXT;
tvi.hItem = hTree;
tvi.pszText = szName;
tvi.cchTextMax = sizeof( szName );
if ( !TreeView_GetItem( hWndTree, &tvi ) )
{
return;
}
char szPath[MAX_PATH];
V_ComposeFileName( pPath, szName, szPath, sizeof( szPath ) );
if ( checkState == 1 && ( bStateIsSame || !tvi.cChildren ) )
{
// add to exclude list
g_ExcludePaths.AddToTail( szPath );
}
if ( !bStateIsSame && tvi.cChildren )
{
// mixed states, must recurse to resolve
HTREEITEM hChild = TreeView_GetChild( hWndTree, hTree );
if ( hChild )
{
ExcludePathsDlg_BuildExcludeList_r( hWndTree, hChild, depth+1, szPath );
}
}
if ( !depth )
{
// only iterate siblings of the parent's child
return;
}
// iterate siblings
HTREEITEM hSibling = hTree;
while ( 1 )
{
hSibling = TreeView_GetNextSibling( hWndTree, hSibling );
if ( !hSibling )
{
break;
}
checkState = -1;
bStateIsSame = ExcludePathsDlg_GetCheckState_r( hWndTree, hSibling, 0, &checkState );
if ( checkState == -1 )
{
break;
}
tvi.hItem = hSibling;
if ( !TreeView_GetItem( hWndTree, &tvi ) )
{
break;
}
V_ComposeFileName( pPath, szName, szPath, sizeof( szPath ) );
if ( checkState == 1 && ( bStateIsSame || !tvi.cChildren ) )
{
// add to exclude list
g_ExcludePaths.AddToTail( szPath );
}
if ( !bStateIsSame && tvi.cChildren )
{
HTREEITEM hChild = TreeView_GetChild( hWndTree, hSibling );
if ( hChild )
{
ExcludePathsDlg_BuildExcludeList_r( hWndTree, hChild, depth+1, szPath );
}
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_SaveChanges
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_SaveChanges( HWND hWnd )
{
g_ExcludePaths.Purge();
HWND hWndTree = GetDlgItem( hWnd, IDC_PATHS_TREE );
ExcludePathsDlg_BuildExcludeList_r( hWndTree, TreeView_GetRoot( hWndTree ), 0, "" );
char szPath[MAX_PATH];
V_ComposeFileName( g_localPath, EXCLUDEPATHS_FILE, szPath, sizeof( szPath ) );
if ( !g_ExcludePaths.Count() )
{
// no exclude paths
unlink( szPath );
}
else
{
FILE *fp = fopen( szPath, "wt" );
if ( !fp )
{
Sys_MessageBox( "Error", "Could not open '%s' for writing\n", szPath );
return;
}
fprintf( fp, "// Auto-Generated by VXConsole - DO NOT MODIFY!\n" );
for ( int i = 0; i < g_ExcludePaths.Count(); i++ )
{
// strip expected root path
const char *pPath = g_ExcludePaths[i].String();
pPath += strlen( ROOT_NAME );
if ( !pPath[0] )
{
// special code for root
fprintf( fp, "*\n" );
break;
}
fprintf( fp, "\"\\%s\"\n", pPath );
}
fclose( fp );
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Populate
//
// Generate a path table.
//-----------------------------------------------------------------------------
void ExcludePathsDlg_Populate( HWND hWnd, bool bRefresh )
{
struct GamePath_t
{
CUtlString pathName;
bool bIsModPath;
};
CUtlVector< GamePath_t > gamePaths;
// can skip the time consuming directory scan, unless forced
if ( bRefresh || !g_PathTable.Count() )
{
g_PathTable.Purge();
for ( int i = 0; i < ARRAYSIZE( g_GameNames ); i++ )
{
char szTargetPath[MAX_PATH];
V_strncpy( szTargetPath, g_localPath, sizeof( szTargetPath ) );
V_AppendSlash( szTargetPath, sizeof( szTargetPath ) );
V_strncat( szTargetPath, g_GameNames[i].pName, sizeof( szTargetPath ) );
GamePath_t gamePath;
gamePath.pathName = szTargetPath;
gamePath.bIsModPath = g_GameNames[i].bIsModPath;
gamePaths.AddToTail( gamePath );
}
// iterate all game paths
for ( int i = 0; i < gamePaths.Count(); i++ )
{
CUtlVector< CUtlString > dirList;
RecurseFileTree_r( g_localPath, gamePaths[i].pathName.String(), 0, dirList, gamePaths[i].bIsModPath );
}
}
// reset the tree
HWND hWndTree = GetDlgItem( hWnd, IDC_PATHS_TREE );
TreeView_DeleteAllItems( hWndTree );
// reset and add root
// only the root is at depth 0
AddItemToTree( hWndTree, ROOT_NAME, 0, false );
// build the tree view
for ( int iIndex = g_PathTable.FirstInorder(); iIndex != g_PathTable.InvalidIndex(); iIndex = g_PathTable.NextInorder( iIndex ) )
{
// due to sorting, number of slashes is depth
const char *pString = g_PathTable[iIndex].pathName.String();
int depth = 0;
for ( int j = 0; j < (int)strlen( pString ); j++ )
{
if ( pString[j] == '\\' )
{
depth++;
}
}
if ( !depth )
{
depth = 1;
}
char szPath[MAX_PATH];
V_FileBase( pString, szPath, sizeof( szPath ) );
AddItemToTree( hWndTree, szPath, depth, g_PathTable[iIndex].bIsModPath );
}
HTREEITEM hTreeRoot = TreeView_GetRoot( hWndTree );
for ( int i = 0; i < g_ExcludePaths.Count(); i++ )
{
HTREEITEM hTree = ExcludePathsDlg_Find_r( hWndTree, hTreeRoot, 0, "", g_ExcludePaths[i].String() );
if ( hTree )
{
ExcludePathsDlg_SetCheckState_r( hWndTree, hTree, 0, true );
}
}
// expand the root node to show state
ExcludePathsDlg_ExpandToShowState_r( hWndTree, hTreeRoot, 0 );
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Setup
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_Setup( HWND hWnd )
{
TreeView_SetBkColor( GetDlgItem( hWnd, IDC_PATHS_TREE ), g_backgroundColor );
CheckDlgButton( hWnd, IDC_PATHS_LINKGAMEDIRS, g_bLinkGameDirs ? BST_CHECKED : BST_UNCHECKED );
// read the exisiting exclude paths
g_ExcludePaths.Purge();
char szFilename[MAX_PATH];
V_ComposeFileName( g_localPath, EXCLUDEPATHS_FILE, szFilename, sizeof( szFilename ) );
if ( Sys_Exists( szFilename ) )
{
Sys_LoadScriptFile( szFilename );
while ( 1 )
{
char *pToken = Sys_GetToken( true );
if ( !pToken || !pToken[0] )
{
break;
}
Sys_StripQuotesFromToken( pToken );
if ( !stricmp( pToken, "*" ) )
{
pToken = "";
}
else if ( pToken[0] == '\\' )
{
pToken++;
}
char szPath[MAX_PATH];
V_ComposeFileName( ROOT_NAME, pToken, szPath, sizeof( szPath ) );
V_FixSlashes( szPath );
g_ExcludePaths.AddToTail( szPath );
}
}
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Proc
//
//-----------------------------------------------------------------------------
BOOL CALLBACK ExcludePathsDlg_Proc( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam )
{
HWND hWndTree;
LPNMHDR lpnmh;
HTREEITEM hTree;
HTREEITEM hTreeRoot;
int checkState;
static bool s_bAllowPopulate = 0;
switch ( message )
{
case WM_INITDIALOG:
ExcludePathsDlg_Setup( hWnd );
s_bAllowPopulate = true;
return TRUE;
case WM_NOTIFY:
lpnmh = (LPNMHDR)lParam;
if ( ( lpnmh->code == NM_CLICK ) && ( lpnmh->idFrom == IDC_PATHS_TREE ) )
{
TVHITTESTINFO ht = {0};
DWORD dwpos = GetMessagePos();
ht.pt.x = GET_X_LPARAM( dwpos );
ht.pt.y = GET_Y_LPARAM( dwpos );
MapWindowPoints( HWND_DESKTOP, lpnmh->hwndFrom, &ht.pt, 1 );
TreeView_HitTest( lpnmh->hwndFrom, &ht );
if ( ht.flags & TVHT_ONITEMSTATEICON )
{
PostMessage( hWnd, UM_CHECKSTATECHANGE, 0, (LPARAM)ht.hItem );
}
}
break;
case UM_CHECKSTATECHANGE:
hWndTree = GetDlgItem( hWnd, IDC_PATHS_TREE );
hTreeRoot = TreeView_GetRoot( hWndTree );
hTree = (HTREEITEM)lParam;
checkState = TreeView_GetCheckState( hWndTree, hTree );
if ( checkState != -1 )
{
TVITEM tvi = { 0 };
char szName[MAX_PATH];
tvi.mask = TVIF_HANDLE | TVIF_PARAM | TVIF_TEXT;
tvi.hItem = hTree;
tvi.pszText = szName;
tvi.cchTextMax = sizeof( szName );
if ( !TreeView_GetItem( hWndTree, &tvi ) )
{
break;
}
int nDepth = LOWORD( tvi.lParam );
bool bIsModPath = HIWORD( tvi.lParam ) != 0;
if ( g_bLinkGameDirs && bIsModPath )
{
// a mod path root is at depth 1
// a child of a mod path (depth > 1), will match all other children in mod paths
// a mod path (depth = 1) will match all other modpaths
ExcludePathsDlg_SetCheckStateLinked_r( hWndTree, hTreeRoot, 0, checkState, nDepth == 1 ? NULL : szName );
}
else
{
ExcludePathsDlg_SetCheckState_r( hWndTree, hTree, 0, checkState );
}
}
break;
case WM_PAINT:
if ( s_bAllowPopulate )
{
// unfortunate, but tree view needs to paint first before its state can be set
// related to checkbox style which doesn't manifest until after WM_INITDIALOG
// stall the initial population until after the first paint
s_bAllowPopulate = false;
PostMessage( hWnd, UM_FIRSTTIMEPOPULATE, 0, 0 );
}
break;
case UM_FIRSTTIMEPOPULATE:
ExcludePathsDlg_Populate( hWnd, false );
break;
case WM_COMMAND:
switch ( LOWORD( wParam ) )
{
case IDC_OK:
ExcludePathsDlg_SaveChanges( hWnd );
EndDialog( hWnd, wParam );
return TRUE;
case IDC_PATHS_RESCAN:
ExcludePathsDlg_Populate( hWnd, true );
return TRUE;
case IDC_PATHS_EXPAND:
// expand from root
hWndTree = GetDlgItem( hWnd, IDC_PATHS_TREE );
ExcludePathsDlg_Expand_r( hWndTree, TreeView_GetRoot( hWndTree ), 0, -1, true );
return TRUE;
case IDC_PATHS_COLLAPSE:
// collapse from root
hWndTree = GetDlgItem( hWnd, IDC_PATHS_TREE );
ExcludePathsDlg_Expand_r( hWndTree, TreeView_GetRoot( hWndTree ), 0, -1, false );
return TRUE;
case IDC_PATHS_LINKGAMEDIRS:
g_bLinkGameDirs = IsDlgButtonChecked( hWnd, IDC_PATHS_LINKGAMEDIRS );
return TRUE;
case IDCANCEL:
case IDC_CANCEL:
EndDialog( hWnd, wParam );
return TRUE;
}
break;
}
return FALSE;
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_LoadConfig
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_LoadConfig()
{
// get our config
Sys_GetRegistryInteger( "ExcludePaths_LinkGameDirs", true, g_bLinkGameDirs );
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_SaveConfig
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_SaveConfig()
{
Sys_SetRegistryInteger( "ExcludePaths_LinkGameDirs", g_bLinkGameDirs );
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Init
//
//-----------------------------------------------------------------------------
bool ExcludePathsDlg_Init()
{
return true;
}
//-----------------------------------------------------------------------------
// ExcludePathsDlg_Open
//
//-----------------------------------------------------------------------------
void ExcludePathsDlg_Open()
{
ExcludePathsDlg_LoadConfig();
int result = DialogBox( g_hInstance, MAKEINTRESOURCE( IDD_EXCLUDEPATHS ), g_hDlgMain, ( DLGPROC )ExcludePathsDlg_Proc );
if ( LOWORD( result ) != IDC_OK )
return;
ExcludePathsDlg_SaveConfig();
}