windows-nt/Source/XPSP1/NT/windows/appcompat/tools/compatadmin/wizard.cpp
2020-09-26 16:20:57 +08:00

1873 lines
59 KiB
C++

#include "compatadmin.h"
#ifndef __WIZARD_H
#include "wizard.h"
#endif
#include "xmldialog.h"
CShimWizard * g_pCurrentWizard = NULL;
#define MAX_DESC_LENGTH 40
#define PAGE_INTRO 0
#define PAGE_APPNAME 1
#define PAGE_LAYERNAME 2
#define PAGE_SELFILES1 3
#define PAGE_DONE 3
#define PAGE_SHIMS 4
#define PAGE_SHIMNAME 5
#define PAGE_SELECTFILES 6
BOOL CShimWizard::BeginWizard(HWND hParent)
{
PROPSHEETPAGE Pages[11];
ZeroMemory(&m_Record,sizeof(m_Record));
CoCreateGuid(&m_Record.guidID);
// Setup wizard variables
g_pCurrentWizard = this;
// Set default application settings
m_uType = TYPE_LAYER;
// begin the wizard
PROPSHEETHEADER Header;
Header.dwSize = sizeof(PROPSHEETHEADER);
Header.dwFlags = PSH_WIZARD97 | PSH_PROPSHEETPAGE | PSH_HEADER;
Header.hwndParent = hParent;
Header.hInstance = g_hInstance;
Header.pszCaption = /*"Create an application fix";//*/MAKEINTRESOURCE(IDS_WIZARD);
Header.nStartPage = 0;
Header.ppsp = Pages;
Header.nPages = 7;
Header.pszbmHeader = MAKEINTRESOURCE(IDB_WIZBMP);
Pages[PAGE_INTRO].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_INTRO].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_INTRO].hInstance = g_hInstance;
Pages[PAGE_INTRO].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD);
Pages[PAGE_INTRO].pfnDlgProc = (DLGPROC)EntryPoint;
Pages[PAGE_INTRO].pszHeaderSubTitle = TEXT("Select method");
Pages[PAGE_APPNAME].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_APPNAME].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_APPNAME].hInstance = g_hInstance;
Pages[PAGE_APPNAME].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD3);
Pages[PAGE_APPNAME].pfnDlgProc = (DLGPROC)GetAppName;
Pages[PAGE_APPNAME].pszHeaderSubTitle = TEXT("Enter an application name");
Pages[PAGE_LAYERNAME].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_LAYERNAME].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_LAYERNAME].hInstance = g_hInstance;
Pages[PAGE_LAYERNAME].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD2);
Pages[PAGE_LAYERNAME].pfnDlgProc = (DLGPROC)SelectLayer;
Pages[PAGE_LAYERNAME].pszHeaderSubTitle = TEXT("Select the files and compatibility mode");
/*
Pages[PAGE_SELFILES1].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_SELFILES1].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_SELFILES1].hInstance = g_hInstance;
Pages[PAGE_SELFILES1].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD4);
Pages[PAGE_SELFILES1].pfnDlgProc = SelectMatching;
Pages[PAGE_SELFILES1].pszHeaderSubTitle = "How would you like to identify the application?";
*/
Pages[PAGE_DONE].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_DONE].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_DONE].hInstance = g_hInstance;
Pages[PAGE_DONE].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARDDONE);
Pages[PAGE_DONE].pfnDlgProc = (DLGPROC)WizardDone;
Pages[PAGE_DONE].pszHeaderSubTitle = TEXT("You have successfully created an application fix");
Pages[PAGE_SHIMS].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_SHIMS].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_SHIMS].hInstance = g_hInstance;
Pages[PAGE_SHIMS].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD5);
Pages[PAGE_SHIMS].pfnDlgProc = (DLGPROC)SelectShims;
Pages[PAGE_SHIMS].pszHeaderSubTitle = TEXT("Select the fixes to apply to the application");
Pages[PAGE_SHIMNAME].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_SHIMNAME].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_SHIMNAME].hInstance = g_hInstance;
Pages[PAGE_SHIMNAME].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD6);
Pages[PAGE_SHIMNAME].pfnDlgProc = (DLGPROC)SelectLayer;
Pages[PAGE_SHIMNAME].pszHeaderSubTitle = TEXT("Select the file to create the fix for");
Pages[PAGE_SELECTFILES].dwSize = sizeof(PROPSHEETPAGE);
Pages[PAGE_SELECTFILES].dwFlags = PSP_USEHEADERSUBTITLE;
Pages[PAGE_SELECTFILES].hInstance = g_hInstance;
Pages[PAGE_SELECTFILES].pszTemplate = MAKEINTRESOURCE(IDD_ADDWIZARD7);
Pages[PAGE_SELECTFILES].pfnDlgProc = (DLGPROC)SelectFiles;
Pages[PAGE_SELECTFILES].pszHeaderSubTitle = TEXT("Select files used for application identification");
if ( 0 < PropertySheet(&Header) ) {
PDBRECORD pRecord = new DBRECORD;
if ( NULL != pRecord ) {
ZeroMemory(pRecord,sizeof(DBRECORD));
pRecord->szEXEName = m_Record.szEXEName;
pRecord->szAppName = m_Record.szAppName;
pRecord->szLayerName = m_Record.szLayerName;
pRecord->guidID = m_Record.guidID;
pRecord->pEntries = m_Record.pEntries;
g_theApp.GetDBLocal().InsertRecord(pRecord);
return TRUE;
}
}
return FALSE;
}
BOOL CALLBACK EntryPoint(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
HWND hParent = GetParent(hDlg);
SetWindowText(hParent,TEXT("Create an application fix"));
if ( TYPE_LAYER == g_pCurrentWizard->m_uType )
SendDlgItemMessage(hDlg,IDC_LAYERS,BM_SETCHECK,BST_CHECKED,0);
if ( TYPE_SHIM == g_pCurrentWizard->m_uType )
SendDlgItemMessage(hDlg,IDC_SHIM,BM_SETCHECK,BST_CHECKED,0);
if ( TYPE_APPHELP == g_pCurrentWizard->m_uType )
SendDlgItemMessage(hDlg,IDC_APPHELP,BM_SETCHECK,BST_CHECKED,0);
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_SETACTIVE:
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS, 0, PSWIZB_NEXT);
break;
case PSN_WIZNEXT:
{
if ( BST_CHECKED == SendDlgItemMessage(hDlg,IDC_LAYERS,BM_GETCHECK,0,0) ) {
if ( g_pCurrentWizard->m_uType != TYPE_LAYER )
g_pCurrentWizard->WipeRecord(TRUE,TRUE,TRUE);
g_pCurrentWizard->m_uType = TYPE_LAYER;
}
if ( BST_CHECKED == SendDlgItemMessage(hDlg,IDC_SHIM,BM_GETCHECK,0,0) ) {
if ( g_pCurrentWizard->m_uType != TYPE_SHIM )
g_pCurrentWizard->WipeRecord(TRUE,TRUE,TRUE);
g_pCurrentWizard->m_uType = TYPE_SHIM;
}
}
break;
}
}
break;
}
return FALSE;
}
#define FRIENDLY_NAME TEXT("My Application Fix")
BOOL CALLBACK GetAppName(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
SendMessage(
GetDlgItem(hDlg,IDC_NAME), // handle to destination window
EM_LIMITTEXT, // message to send
(WPARAM) LIMIT_APP_NAME, // text length
(LPARAM) 0
);
SHAutoComplete(GetDlgItem(hDlg,IDC_NAME), AUTOCOMPLETE);
if ( 0 == g_pCurrentWizard->m_Record.szAppName.Length() )
g_pCurrentWizard->m_Record.szAppName = FRIENDLY_NAME;
SetDlgItemText(hDlg,IDC_NAME, g_pCurrentWizard->m_Record.szAppName);
if ( g_pCurrentWizard->m_Record.szAppName == FRIENDLY_NAME )
SendMessage(GetDlgItem(hDlg,IDC_NAME),EM_SETSEL,0,-1);
// Force proper Next button state.
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_SETACTIVE:
{
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
if ( TYPE_LAYER == g_pCurrentWizard->m_uType )
SetWindowText(GetDlgItem(hDlg,IDC_TITLE),TEXT("Enter the name of the application to create a compatibility layer for"));
else
SetWindowText(GetDlgItem(hDlg,IDC_TITLE),TEXT("Enter the name of the application to create a fix for"));
}
break;
case PSN_WIZNEXT:
{
TCHAR szTemp[MAX_STRING_SIZE];
GetDlgItemText(hDlg,IDC_NAME,szTemp,MAX_STRING_SIZE);
CSTRING::Trim(szTemp);
g_pCurrentWizard->m_Record.szAppName = szTemp;
if ( TYPE_SHIM == g_pCurrentWizard->m_uType ) {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD6);
return IDD_ADDWIZARD6;
}
}
break;
}
}
break;
case WM_COMMAND:
switch ( LOWORD(wParam) ) {
case IDC_NAME:
if ( EN_CHANGE == HIWORD(wParam) ) {
TCHAR szText[MAX_PATH_BUFFSIZE];
GetWindowText(GetDlgItem(hDlg,IDC_NAME),szText,MAX_PATH);
BOOL bEnable = ( CSTRING::Trim(szText) > 0) ? TRUE:FALSE;
DWORD dwFlags = PSWIZB_BACK;
if ( bEnable )
dwFlags |= PSWIZB_NEXT;
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS,0, dwFlags);
}
break;
}
}
return FALSE;
}
BOOL CALLBACK SelectLayer(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
SendMessage(
GetDlgItem(hDlg,IDC_NAME), // handle to destination window
EM_LIMITTEXT, // message to send
(WPARAM) MAX_PATH, // text length
(LPARAM) 0
);
SetDlgItemText(hDlg,IDC_NAME,g_pCurrentWizard->m_szLongName);
SHAutoComplete(GetDlgItem(hDlg,IDC_NAME), AUTOCOMPLETE);
// Force proper Next button state.
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_SETACTIVE:
{
if ( TYPE_LAYER == g_pCurrentWizard->m_uType ) {
int nSelIndex = -1;
// Remove all strings.
//g_pCurrentWizard->WipeRecord(TRUE,TRUE,FALSE);
while ( CB_ERR != SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_DELETESTRING,0,0) );
// Re-add the strings.
PDBLAYER pWalk = g_theApp.GetDBGlobal().m_pLayerList;
for ( int iLoop = 0 ; iLoop <= 1 ; ++ iLoop ){
//
// Do both for the local and the global Databases
//
while ( NULL != pWalk ) {
if ( CB_ERR == SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_FINDSTRINGEXACT,0,(LPARAM)(LPCTSTR)pWalk->szLayerName) ) {
int nIndex;
nIndex = SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_ADDSTRING,0,(LPARAM)(LPTSTR)pWalk->szLayerName);
if ( CB_ERR != nIndex ) {
SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_SETITEMDATA,nIndex,(LPARAM)pWalk);
// Select this index if it's the current layer name.
/*
if (0 == lstrcmp(g_pCurrentWizard->m_Record.szLayerName,(LPSTR)pWalk->szLayerName))
nSelIndex = nIndex;
*/
}
}
pWalk = pWalk->pNext;
}
pWalk = g_theApp.GetDBLocal().m_pLayerList;
} //for
nSelIndex = SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_FINDSTRINGEXACT,0,(LPARAM)(LPCTSTR)g_pCurrentWizard->m_Record.szLayerName);
if ( -1 != nSelIndex )
SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_SETCURSEL,nSelIndex,0);
}
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
}
break;
case PSN_WIZNEXT:
{
TCHAR szTemp[MAX_STRING_SIZE];
GetDlgItemText(hDlg,IDC_NAME,szTemp,MAX_PATH);
CSTRING::Trim(szTemp);
HANDLE hFile = CreateFile (szTemp,0,0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
if ( INVALID_HANDLE_VALUE == hFile ) {
MessageBox(hDlg,TEXT("Unable to locate specified file"),TEXT("Invalid file name"),MB_OK);
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,-1);
return -1;
}
CloseHandle(hFile);
DWORD dwExeType;
GetBinaryType(szTemp, &dwExeType);
BOOL bExe = FALSE;// Is this an .exe file
CSTRING strTemp = szTemp;
if ( strTemp.EndsWith(TEXT(".exe")) ) bExe = TRUE;
//
// Check if this is "shimmable"
//
CSTRING msg;
if ( (dwExeType & SCS_DOS_BINARY) && bExe )
msg = TEXT("This is a DOS Application\n");
else if ( (dwExeType & SCS_WOW_BINARY) && bExe)
msg = TEXT("This is a 16 bit Windows Application\n");
else if ((dwExeType & SCS_PIF_BINARY) && bExe )
msg = TEXT("This is a PIF Binary Application\n");
else if ( (dwExeType & SCS_POSIX_BINARY) && bExe)
msg = msg = TEXT("This is a POSIX Binary Application\n");
else if ( (dwExeType & SCS_OS216_BINARY) && bExe)
msg = TEXT("This is a OS2 16 bit Application\n");
if (msg.Length() > 0) {
//
//So this application cannot be fixed
//
MessageBox(hDlg,
msg.strcat( TEXT("The fix may not get applied properly for this application") ),
TEXT("Warning!"),
MB_ICONWARNING
);
//SetWindowLongPtr(hDlg,DWLP_MSGRESULT,-1);
//return -1;
}
////check - out
if (strTemp != g_pCurrentWizard->m_szLongName) {
//
//The file name was changed. Either this is the first time that the user has come here, or has moved back and changed the file name
// Remove all the mathcing info
g_pCurrentWizard->WipeRecord(TRUE, FALSE, FALSE);
}
////
g_pCurrentWizard->m_szLongName = szTemp;
g_pCurrentWizard->m_Record.szEXEName = g_pCurrentWizard->ShortFile(g_pCurrentWizard->m_szLongName);
if ( TYPE_LAYER == g_pCurrentWizard->m_uType ) {
int nIndex;
nIndex = SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_GETCURSEL,0,0);
SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_GETLBTEXT,nIndex,(LPARAM)szTemp);
g_pCurrentWizard->m_Record.szLayerName = szTemp;
} else {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD5);
return IDD_ADDWIZARD6;
}
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD7);
return IDD_ADDWIZARD7;
}
break;
case PSN_WIZBACK:
{
TCHAR szTemp[MAX_STRING_SIZE];
GetDlgItemText(hDlg,IDC_NAME,szTemp,MAX_STRING_SIZE);
CSTRING::Trim(szTemp);
g_pCurrentWizard->m_szLongName = szTemp;
if ( TYPE_LAYER == g_pCurrentWizard->m_uType ) {
int nIndex;
nIndex = SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_GETCURSEL,0,0);
SendDlgItemMessage(hDlg,IDC_LAYERLIST,CB_GETLBTEXT,nIndex,(LPARAM)szTemp);
g_pCurrentWizard->m_Record.szLayerName = szTemp;
} else {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD3);
return IDD_ADDWIZARD3;
}
}
break;
}
}
break;
case WM_COMMAND:
switch ( LOWORD(wParam) ) {
case IDC_LAYERLIST:
{
if ( CBN_SELCHANGE == HIWORD(wParam) )
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
}
break;
case IDC_NAME:
if ( EN_CHANGE == HIWORD(wParam) ) {
TCHAR szText[MAX_PATH_BUFFSIZE];
GetWindowText(GetDlgItem(hDlg,IDC_NAME),szText,MAX_PATH);
BOOL bEnable = (CSTRING::Trim(szText) > 0) ? TRUE:FALSE;
DWORD dwFlags = PSWIZB_BACK;
if ( bEnable )
dwFlags |= PSWIZB_NEXT;
HWND hLayer = GetDlgItem(hDlg,IDC_LAYERLIST);
if ( NULL != hLayer ) {
// A layer must be selected as well.
int nSel = SendMessage(hLayer,CB_GETCURSEL,0,0);
if ( CB_ERR == nSel )
dwFlags &= ~PSWIZB_NEXT;
}
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS,0, dwFlags);
}
break;
case IDC_BROWSE:
{
CSTRING szFilename;
HWND hwndFocus = GetFocus();
if ( g_theApp.GetFilename(TEXT("Find executable"),TEXT("EXE File (*.EXE)\0*.EXE\0All files (*.*)\0*.*\0\0"),TEXT(""),TEXT("EXE"),OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST,TRUE,szFilename) ) {
g_pCurrentWizard->m_Record.szEXEName = g_pCurrentWizard->ShortFile(szFilename);
SetDlgItemText(hDlg, IDC_NAME, szFilename);
SendMessage(hDlg,WM_COMMAND,MAKEWPARAM(IDC_NAME,EN_CHANGE),0);
}
SetFocus( hwndFocus );
}
break;
}
break;
}
return FALSE;
}
BOOL CALLBACK SelectMatching(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
SendDlgItemMessage(hDlg,IDC_GENERATE,BM_SETCHECK,BST_CHECKED,0);
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_SETACTIVE:
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS, 0, PSWIZB_BACK | PSWIZB_NEXT);
break;
case PSN_WIZNEXT:
{
if ( BST_CHECKED == SendDlgItemMessage(hDlg,IDC_GENERATE,BM_GETCHECK,0,0) ) {
HCURSOR hRestore;
CSTRING szFile = g_pCurrentWizard->m_szLongName;
LPTSTR szWalk = _tcsrchr( (LPTSTR)szFile, TEXT('\\'));
//
// current directory extraction from the szFile
//
if (NULL == szWalk) {
// ?
} else {
*szWalk = 0;
}
SetCurrentDirectory(szFile);
g_pCurrentWizard->m_bManualMatch = FALSE;
hRestore = SetCursor(LoadCursor(NULL,IDC_WAIT));
g_pCurrentWizard->GrabMatchingInfo();
SetCursor(hRestore);
}
if ( BST_CHECKED == SendDlgItemMessage(hDlg,IDC_MANUAL,BM_GETCHECK,0,0) ) {
WIN32_FIND_DATA Data;
HANDLE hFile;
PMATCHENTRY pMatch = NULL;
g_pCurrentWizard->m_bManualMatch = TRUE;
hFile = FindFirstFile(g_pCurrentWizard->m_szLongName,&Data);
if ( INVALID_HANDLE_VALUE != hFile ) {
g_pCurrentWizard->AddMatchFile(&pMatch,g_pCurrentWizard->m_szLongName);
g_pCurrentWizard->GetFileAttributes(pMatch);
if ( !g_pCurrentWizard->InsertMatchingInfo(pMatch) )
delete pMatch;
//pMatch->Entry.pNext = g_pCurrentWizard->m_Record.pEntries;
//g_pCurrentWizard->m_Record.pEntries = (PDBENTRY) pMatch;
}
}
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD7);
return IDD_ADDWIZARD7;
}
break;
case PSN_WIZBACK:
{
if ( TYPE_LAYER == g_pCurrentWizard->m_uType ) {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD2);
return IDD_ADDWIZARD2;
} else
if ( TYPE_SHIM == g_pCurrentWizard->m_uType ) {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD5);
return IDD_ADDWIZARD5;
}
}
break;
}
}
break;
}
return FALSE;
}
BOOL CALLBACK WizardDone(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_WIZBACK:
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD7);
return IDD_ADDWIZARD7;
case PSN_SETACTIVE:
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS, 0, PSWIZB_BACK | PSWIZB_FINISH);
}
}
break;
case WM_COMMAND:
switch ( LOWORD(wParam) ) {
case IDC_VIEWXML:
{
CXMLDialog XML;
XML.BeginXMLView(&g_pCurrentWizard->m_Record,hDlg,FALSE,FALSE,TRUE);
}
break;
case IDC_TESTRUN:
{
HWND hndFocus = GetFocus();
g_theApp.TestRun(&g_pCurrentWizard->m_Record,&g_pCurrentWizard->m_szLongName,NULL,hDlg);
SetFocus(hndFocus);
}
break;
}
break;
}
return FALSE;
}
void SelectShims_TreeDoubleClicked(HWND hDlg);
BOOL CALLBACK SelectShims(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
TVINSERTSTRUCT Item;
PSHIMDESC pWalk = g_theApp.GetDBGlobal().m_pShimList;
Item.hParent = TVI_ROOT;
Item.hInsertAfter = TVI_LAST;
Item.item.mask = TVIF_TEXT | TVIF_PARAM;
while ( NULL != pWalk ) {
if ( pWalk->bGeneral ) {
PSHIMENTRY pNew = new SHIMENTRY;
if ( NULL != pNew ) {
pNew->Entry.uType = ENTRY_SHIM;
pNew->Entry.uIconID = 0;
pNew->Entry.pNext = NULL;
pNew->szShimName = pWalk->szShimName;
pNew->szCmdLine = pWalk->szShimCommandLine;
pNew->pDesc = pWalk;
Item.item.pszText = (LPTSTR) pWalk->szShimName;
Item.item.cchTextMax = lstrlen(Item.item.pszText);
Item.item.lParam = (LPARAM) pNew;
TreeView_InsertItem(GetDlgItem(hDlg,IDC_SHIMLIST),&Item);
}
}
pWalk = pWalk->pNext;
}
SetTimer(hDlg,0,100,NULL);
}
break;
case WM_COMMAND:
{
switch ( LOWORD(wParam) ) {
case IDC_CLEARALL:
{
HTREEITEM hItem;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
hItem = TreeView_GetRoot(hTree);
while ( NULL != hItem ) {
TVITEM Item;
Item.mask = TVIF_STATE;
Item.hItem = hItem;
TreeView_GetItem(hTree,&Item);
if ( 0 != (Item.state & 0x2000) ) {
Item.state &= 0xFFFFDFFF;
Item.state |= 0x1000;
TreeView_SetItemState(hTree,hItem,Item.state,0xFFFFFFFF);
}
hItem = TreeView_GetNextSibling(hTree,hItem);
}
// Recount
SetTimer(hDlg,0,100,NULL);
}
break;
}
}
break;
case WM_TIMER:
{
UINT uTotal = 0;
UINT uSelected = 0;
HTREEITEM hItem;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
CSTRING szText;
KillTimer(hDlg,0);
// Count the selected shims
hItem = TreeView_GetRoot(hTree);
while ( NULL != hItem ) {
TVITEM Item;
Item.mask = TVIF_STATE;
Item.hItem = hItem;
TreeView_GetItem(hTree,&Item);
if ( 0 != (Item.state & 0x2000) )
++uSelected;
++uTotal;
hItem = TreeView_GetNextSibling(hTree,hItem);
}
szText.sprintf(TEXT("Selected %d of %d"),uSelected,uTotal);
SetWindowText(GetDlgItem(hDlg,IDC_STATUS),(LPCTSTR)szText);
DWORD dwFlags = PSWIZB_BACK | PSWIZB_NEXT;
if ( 0 == uSelected )
dwFlags &= ~PSWIZB_NEXT;
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS,0, dwFlags);
}
break;
case WM_DESTROY:
{
HTREEITEM hItem;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
hItem = TreeView_GetRoot(hTree);
while ( NULL != hItem ) {
TVITEM Item;
Item.mask = TVIF_STATE | TVIF_PARAM;
Item.hItem = hItem;
TreeView_GetItem(hTree,&Item);
PSHIMENTRY pEntry = (PSHIMENTRY) Item.lParam;
PDBENTRY pWalk = g_pCurrentWizard->m_Record.pEntries;
while ( NULL != pWalk ) {
if ( pWalk == (PDBENTRY)pEntry )
break;
pWalk = pWalk->pNext;
}
if ( NULL == pWalk )
delete pEntry;
hItem = TreeView_GetNextSibling(hTree,hItem);
}
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case NM_RETURN:{
SelectShims_TreeDoubleClicked(hDlg);
return TRUE;
}
case PSN_SETACTIVE:
SetTimer(hDlg,0,100,NULL);
break;
case NM_DBLCLK:
{
SelectShims_TreeDoubleClicked(hDlg);
}//case NM_DBLCLK:
break;
case NM_CLICK:
{
TVHITTESTINFO ht;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
GetCursorPos(&ht.pt);
ScreenToClient(hTree, &ht.pt);
TreeView_HitTest(hTree,&ht);
if ( 0 != ht.hItem )
TreeView_SelectItem(hTree,ht.hItem);
SetTimer(hDlg,0,100,NULL);
}
break;
case PSN_WIZBACK:
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD6);
return IDD_ADDWIZARD6;
case PSN_WIZNEXT:
{
// Build the shim list.
HTREEITEM hItem;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
// Wipe shims and layers, but not matching info.
g_pCurrentWizard->WipeRecord(FALSE,TRUE,TRUE);
hItem = TreeView_GetRoot(hTree);
while ( NULL != hItem ) {
TVITEM Item;
Item.mask = TVIF_STATE | TVIF_PARAM;
Item.hItem = hItem;
TreeView_GetItem(hTree,&Item);
PSHIMENTRY pEntry = (PSHIMENTRY) Item.lParam;
pEntry->Entry.pNext = NULL;
if ( 0 != (Item.state & 0x2000) ) {
pEntry->Entry.pNext = g_pCurrentWizard->m_Record.pEntries;
g_pCurrentWizard->m_Record.pEntries = (PDBENTRY) pEntry;
}
hItem = TreeView_GetNextSibling(hTree,hItem);
}
//SetWindowLong(hDlg,DWL_MSGRESULT,IDD_ADDWIZARD4);
//return IDD_ADDWIZARD4;
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD7);
return IDD_ADDWIZARD7;
}
break;
case TVN_KEYDOWN:{
LPNMTVKEYDOWN pTvkd = (LPNMTVKEYDOWN) lParam ;
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
if (pTvkd->wVKey == VK_SPACE ) {
if (TreeView_GetSelection(hTree) != NULL ){
SetTimer(hDlg,0,100,NULL);
}
}
break;
}
case TVN_SELCHANGED:
{
LPNMTREEVIEW pTree = (LPNMTREEVIEW) lParam;
PSHIMENTRY pEntry = (PSHIMENTRY) pTree->itemOld.lParam;
TCHAR szCmdLine[MAX_PATH_BUFFSIZE];
/* K BUG why is this required !
if ( NULL != pEntry ) {
GetWindowText(GetDlgItem(hDlg,IDC_CMDLINE),szCmdLine,MAX_PATH);
pEntry->szCmdLine = szCmdLine;
}
*/
pEntry = (PSHIMENTRY) pTree->itemNew.lParam;
PSHIMDESC pDesc = (PSHIMDESC) pEntry->pDesc;
SetWindowText(GetDlgItem(hDlg,IDC_SHIMDESC),(LPCTSTR) pDesc->szShimDesc);
SetWindowText(GetDlgItem(hDlg,IDC_CMDLINE),(LPCTSTR) pEntry->szCmdLine);
}
break;
}
}
break;
}
return FALSE;
}
BOOL CALLBACK SelectFiles(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch ( uMsg ) {
case WM_INITDIALOG:
{
PostMessage(hDlg,WM_USER+1024,0,0);
}
break;
case WM_USER+1024:
{
PDBENTRY pWalk;
TreeView_DeleteAllItems(GetDlgItem(hDlg,IDC_FILELIST));
pWalk = g_pCurrentWizard->m_Record.pEntries;
TVINSERTSTRUCT Item;
ZeroMemory(&Item,sizeof(TVINSERTSTRUCT));
Item.hParent = TVI_ROOT;
Item.hInsertAfter = TVI_LAST;
Item.item.mask = TVIF_TEXT | TVIF_PARAM;
Item.item.lParam = (LPARAM) NULL;
Item.item.pszText = g_pCurrentWizard->m_Record.szEXEName;
Item.item.cchTextMax = lstrlen(Item.item.pszText);
TreeView_InsertItem(GetDlgItem(hDlg,IDC_FILELIST),&Item);
while ( NULL != pWalk ) {
if ( ENTRY_MATCH == pWalk->uType ) {
TVINSERTSTRUCT Item;
PMATCHENTRY pMatch = (PMATCHENTRY) pWalk;
if ( pMatch->szMatchName != TEXT("*") ) {
ZeroMemory(&Item,sizeof(TVINSERTSTRUCT));
Item.hParent = TVI_ROOT;
Item.hInsertAfter = TVI_LAST;
Item.item.mask = TVIF_TEXT | TVIF_PARAM;
Item.item.lParam = (LPARAM) pMatch;
Item.item.pszText = pMatch->szMatchName;
Item.item.cchTextMax = lstrlen(Item.item.pszText);
TreeView_InsertItem(GetDlgItem(hDlg,IDC_FILELIST),&Item);
}
}
pWalk = pWalk->pNext;
}
}
break;
case WM_NOTIFY:
{
NMHDR * pHdr = (NMHDR *) lParam;
switch ( pHdr->code ) {
case PSN_SETACTIVE:
{
SendMessage(GetParent(hDlg),PSM_SETWIZBUTTONS, 0, PSWIZB_BACK | PSWIZB_NEXT);
// Force refresh of files in list.
PostMessage(hDlg,WM_USER+1024,0,0);
}
break;
case PSN_WIZBACK:
{
PMATCHENTRY pWalk = (PMATCHENTRY) g_pCurrentWizard->m_Record.pEntries;
PMATCHENTRY pPrev;
CSTRING szFile = g_pCurrentWizard->m_szLongName;
szFile.ShortFilename();
// Remove the matching info for the current file if it exists. Otherwise,
// it's possible that if the file is changed, we'll have bogus information
// about it.
while ( NULL != pWalk ) {
if ( ENTRY_MATCH == pWalk->Entry.uType )
if ( pWalk->szMatchName == szFile || pWalk->szMatchName == TEXT("*") ) {
// Remove this entry.
if ( pWalk == (PMATCHENTRY) g_pCurrentWizard->m_Record.pEntries )
g_pCurrentWizard->m_Record.pEntries = g_pCurrentWizard->m_Record.pEntries->pNext;
else
pPrev->Entry.pNext = pWalk->Entry.pNext;
delete pWalk;
break;
}
pPrev = pWalk;
pWalk = (PMATCHENTRY) pWalk->Entry.pNext;
}
if ( TYPE_LAYER == g_pCurrentWizard->m_uType ) {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD2);
return IDD_ADDWIZARD2;
} else if ( TYPE_SHIM == g_pCurrentWizard->m_uType ) {
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARD5);
return IDD_ADDWIZARD5;
}
else if (TYPE_APPHELP == g_pCurrentWizard->m_uType ) {
return IDD_APPHELP1;
}
}
break;
case PSN_WIZNEXT:
{
PMATCHENTRY pMatch = NULL;
// Add self. Self should always be included here.
g_pCurrentWizard->AddMatchFile(&pMatch,g_pCurrentWizard->m_szLongName);
g_pCurrentWizard->GetFileAttributes(pMatch);
if ( !g_pCurrentWizard->InsertMatchingInfo(pMatch) )
delete pMatch;
//pMatch->Entry.pNext = g_pCurrentWizard->m_Record.pEntries;
//g_pCurrentWizard->m_Record.pEntries = (PDBENTRY) pMatch;
if (TYPE_APPHELP == g_pCurrentWizard->m_uType)
return TRUE;
SetWindowLongPtr(hDlg,DWLP_MSGRESULT,IDD_ADDWIZARDDONE);
}
return IDD_ADDWIZARDDONE;
}
}
break;
case WM_COMMAND:
switch ( LOWORD(wParam) ) {
case IDC_GENERATE:
{
HCURSOR hRestore;
CSTRING szFile = g_pCurrentWizard->m_szLongName;
LPTSTR szWalk = (LPTSTR) szFile + szFile.Length() - 1;
while ( TEXT('\\') != *szWalk )
--szWalk;
++szWalk;
*szWalk = 0;
SetCurrentDirectory(szFile);
g_pCurrentWizard->m_bManualMatch = FALSE;
hRestore = SetCursor(LoadCursor(NULL,IDC_WAIT));
g_pCurrentWizard->GrabMatchingInfo();
SetCursor(hRestore);
PostMessage(hDlg,WM_USER+1024,0,0);
}
break;
case IDC_ADDFILES:
{
CSTRING szFilename;
HWND hwndFocus = GetFocus();
if ( g_theApp.GetFilename(TEXT("Find Matching File"),TEXT("EXE File (*.EXE)\0*.EXE\0All Files(*.*)\0*.*\0\0"),TEXT(""),TEXT(""),OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST,TRUE,szFilename) ) {
CSTRING szCheck = szFilename;
szCheck.ShortFilename();
PDBENTRY pWalk;
pWalk = g_pCurrentWizard->m_Record.pEntries;
while ( NULL != pWalk ) {
if ( ENTRY_MATCH == pWalk->uType ) {
PMATCHENTRY pTest = (PMATCHENTRY) pWalk;
CSTRING szShort = pTest->szMatchName;
szShort.ShortFilename();
if ( szShort == szCheck ) {
MessageBox(hDlg,TEXT("This file is already being used for matching information. To update, please remove and re-add it."),TEXT("File matching error"),MB_OK);
return FALSE;
}
}
pWalk = pWalk->pNext;
}
//WIN32_FIND_DATA Data;
//HANDLE hFile;
//hFile = FindFirstFile(szFilename,&Data);
//if (INVALID_HANDLE_VALUE != hFile)
{
PMATCHENTRY pMatch = NULL;
g_pCurrentWizard->AddMatchFile(&pMatch,szFilename);
g_pCurrentWizard->GetFileAttributes(pMatch);
if ( !g_pCurrentWizard->InsertMatchingInfo(pMatch) )
delete pMatch;
//pMatch->Entry.pNext = g_pCurrentWizard->m_Record.pEntries;
//g_pCurrentWizard->m_Record.pEntries = (PDBENTRY) pMatch;
//FindClose(hFile);
PostMessage(hDlg,WM_USER+1024,0,0);
}
}
SetFocus( hwndFocus );
}
break;
case IDC_REMOVEALL:
{
PDBENTRY pWalk;
PDBENTRY pHold;
pWalk = g_pCurrentWizard->m_Record.pEntries;
while ( NULL != pWalk ) {
if ( ENTRY_MATCH == pWalk->uType ) {
if ( g_pCurrentWizard->m_Record.pEntries == pWalk ) {
g_pCurrentWizard->m_Record.pEntries = pWalk->pNext;
pHold = g_pCurrentWizard->m_Record.pEntries;
} else
pHold->pNext = pWalk->pNext;
delete pWalk;
pWalk = pHold;
} else {
pHold = pWalk;
pWalk = pWalk->pNext;
}
}
PostMessage(hDlg,WM_USER+1024,0,0);
}
break;
case IDC_REMOVEFILES:
{
HTREEITEM hItem = TreeView_GetSelection(GetDlgItem(hDlg,IDC_FILELIST));
if ( NULL != hItem ) {
TVITEM Item;
Item.mask = TVIF_PARAM;
Item.hItem = hItem;
TreeView_GetItem(GetDlgItem(hDlg,IDC_FILELIST),&Item);
PDBENTRY pWalk;
PDBENTRY pHold;
if ( NULL == Item.lParam ) {
MessageBeep(MB_OK);
MessageBox(NULL,TEXT("This file is required for file matching"),TEXT("Matching error"),MB_OK);
break;
}
pWalk = g_pCurrentWizard->m_Record.pEntries;
while ( NULL != pWalk ) {
if ( pWalk == (PDBENTRY) Item.lParam )
break;
pHold = pWalk;
pWalk = pWalk->pNext;
}
if ( pWalk == g_pCurrentWizard->m_Record.pEntries )
g_pCurrentWizard->m_Record.pEntries = pWalk->pNext;
else
pHold->pNext = pWalk->pNext;
delete pWalk;
PostMessage(hDlg,WM_USER+1024,0,0);
}
}
break;
}
break;
}
return FALSE;
}
void CShimWizard::WipeRecord(BOOL bMatching, BOOL bShims, BOOL bLayer, BOOL bAppHelp)
{
//BUGBUG : Deletion not done correctly.
//
// Matching files are deleted, shims are not ??
//
PDBENTRY pWalk = m_Record.pEntries;
PDBENTRY pPrev = pWalk;//prefast
while ( NULL != pWalk ) {
PDBENTRY pHold = pWalk->pNext;
BOOL bRemove = FALSE;
if ( ENTRY_SHIM == pWalk->uType && bShims )
bRemove = TRUE;
else
if ( ENTRY_MATCH == pWalk->uType && bMatching ) {
bRemove = TRUE;
delete pWalk;
}
#ifdef _DEBUG
if ( bRemove ) {
if ( m_Record.pEntries != pWalk )
if ( pPrev == pHold )
__asm int 3;
}
#endif
if ( bRemove ) {
if ( m_Record.pEntries == pWalk )
m_Record.pEntries = pHold;
else
pPrev->pNext = pHold;
} else
pPrev = pWalk;
pWalk = pHold;
}
if ( bLayer )
m_Record.szLayerName = TEXT("");
}
typedef struct tagATTRINFO2 {
TAG tAttrID; // tag for this attribute (includes type)
DWORD dwFlags; // flags : such as "not avail" or "not there yet"
union { // anonymous union with values
ULONGLONG ullAttr; // QWORD value (TAG_TYPE_QWORD)
DWORD dwAttr; // DWORD value (TAG_TYPE_DWORD)
TCHAR* lpAttr; // WCHAR* value (TAG_TYPE_STRINGREF)
};
} ATTRINFO2, *PATTRINFO2;
void CShimWizard::GetFileAttributes(PMATCHENTRY pNew)
{
PATTRINFO2 pAttribs;
DWORD dwAttribCount;
CSTRING szFile = pNew->szFullName;
BOOL bIs16Bit = FALSE;
/*
DWORD dwType;
GetBinaryType(szFile,&dwType);
if (SCS_WOW_BINARY == dwType)
bIs16Bit = TRUE;
*/
if ( szFile.Length() == 0 ) {
MEM_ERR;
return;
}
if ( SdbGetFileAttributes((LPCTSTR)szFile,(PATTRINFO *)&pAttribs,&dwAttribCount) ) {
UINT uCount;
for ( uCount=0; uCount<dwAttribCount; ++uCount ) {
if ( 0 != (pAttribs[uCount].dwFlags & ATTRIBUTE_AVAILABLE) ) {
switch ( pAttribs[uCount].tAttrID ) {
case TAG_COMPANY_NAME:
pNew->szCompanyName = (LPCTSTR)pAttribs[uCount].lpAttr;
break;
case TAG_PRODUCT_VERSION:
pNew->szProductVersion = (LPCTSTR)pAttribs[uCount].lpAttr;
break;
case TAG_FILE_DESCRIPTION:
pNew->szDescription = (LPCTSTR)pAttribs[uCount].lpAttr;
break;
case TAG_FILE_VERSION:
pNew->szFileVersion = (LPCTSTR)pAttribs[uCount].lpAttr;
break;
case TAG_CHECKSUM:
pNew->dwChecksum = pAttribs[uCount].dwAttr;
break;
case TAG_BIN_FILE_VERSION:
{
if ( !bIs16Bit )
pNew->FileVersion.QuadPart = pAttribs[uCount].ullAttr;
}
break;
case TAG_BIN_PRODUCT_VERSION:
{
if ( !bIs16Bit )
pNew->ProductVersion.QuadPart = pAttribs[uCount].ullAttr;
}
break;
}
}
}
SdbFreeFileAttributes((PATTRINFO)pAttribs);
}
}
void CShimWizard::AddMatchFile(PPMATCHENTRY ppHead, CSTRING & szFilename)
{
PMATCHENTRY pNew;
PMATCHENTRY pWalk;
PMATCHENTRY pHold;
BOOL bInsertHead=FALSE;
CSTRING szFile; // = RelativePath();
//TCHAR szCurrentPath[MAX_PATH_BUFFSIZE];
pNew = new MATCHENTRY;
if ( NULL == pNew )
return;
ZeroMemory(pNew,sizeof(MATCHENTRY));
//GetCurrentDirectory(MAX_PATH,szCurrentPath);
//if (lstrlen(szCurrentPath) == 3)
//szCurrentPath[2] = 0;
//szFile.sprintf("%s\\%s",szCurrentPath,pData->cFileName);
//szFile = szCurrentPath;
///szFile.strcat(pData->cFileName);
//pNew->szFullName.sprintf("%s\\%s",szCurrentPath,pData->cFileName);
pNew->szFullName = szFilename;
//szFile.strcat(pData->cFileName);
szFile = pNew->szFullName;
szFile.RelativeFile(m_szLongName);
pNew->Entry.uType = ENTRY_MATCH;
pNew->szMatchName = szFile;
HANDLE hFile = CreateFile((LPCTSTR) szFilename,GENERIC_READ,0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
if ( INVALID_HANDLE_VALUE != hFile ) {
//pNew->dwSize = pData->nFileSizeLow;
pNew->dwSize = GetFileSize(hFile,NULL);
CloseHandle(hFile);
}
pWalk = *ppHead;
// Walk the list to determine where to insert it.
while ( NULL != pWalk ) {
if ( pWalk->dwSize < pNew->dwSize )
break;
pHold = pWalk;
pWalk = (PMATCHENTRY) pWalk->Entry.pNext;
}
// Insert it into the head if the head is NULL, OR if this file is
// the same as the file being fixed.
CSTRING szShort = szFilename;
if ( szShort.Length() == 0 ) {
MEM_ERR;
return;
}
szShort.ShortFilename();
if ( NULL == *ppHead || 0 == lstrcmpi(szShort,m_Record.szEXEName) ) {//0 == lstrcmpi(pData->cFileName,m_Record.szEXEName))
bInsertHead = TRUE;
//if (0 == lstrcmpi(pData->cFileName,m_Record.szEXEName))
if ( 0 == lstrcmpi(szShort,m_Record.szEXEName) )
pNew->szMatchName = TEXT("*");
} else
if ( *ppHead == pWalk )
bInsertHead = TRUE;
if ( NULL != pWalk && pWalk->szMatchName == pNew->szMatchName ) {
// Duplicate here. Refuse.
delete pNew;
return;
}
if ( bInsertHead ) {
if ( NULL != *ppHead && 0 == lstrcmpi(pNew->szMatchName,TEXT("*")) ) {
// If the file to fix has aleady been added at the head, special
// case this insert.
pNew->Entry.pNext = (*ppHead)->Entry.pNext;
(*ppHead)->Entry.pNext = (PDBENTRY) pNew;
} else {
// Standard head insert.
pNew->Entry.pNext = (PDBENTRY) *ppHead;
*ppHead = pNew;
}
} else {
pNew->Entry.pNext = (PDBENTRY) pWalk;
pHold->Entry.pNext =(PDBENTRY) pNew;
}
}
void CShimWizard::WalkDirectory(PMATCHENTRY * ppHead, LPCTSTR szDirectory, int nDepth)
{
HANDLE hFile;
WIN32_FIND_DATA Data;
TCHAR szCurrentDir[MAX_PATH_BUFFSIZE] = TEXT("");
int nFiles=0;
if ( 2 <= nDepth )
return;
CSTRING szShortName = m_szLongName;
szShortName.ShortFilename();
// Save the current directory
GetCurrentDirectory(MAX_PATH,szCurrentDir);
// Set to the new directory
SetCurrentDirectory(szDirectory);
// Generate automated matching file information.
hFile = FindFirstFile(TEXT("*.*"),&Data);
if ( INVALID_HANDLE_VALUE == hFile ) {
SetCurrentDirectory(szCurrentDir);
return;
}
do {
if ( 0 == (Data.dwFileAttributes & (FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM)) )
if ( FILE_ATTRIBUTE_DIRECTORY == (Data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ) {
if ( TEXT('.') != Data.cFileName[0] )
WalkDirectory(ppHead,Data.cFileName,nDepth+1);
} else {
++nFiles;
if ( nFiles >= 100 )
break;
if ( 0 != lstrcmpi(szShortName,Data.cFileName) )
if ( 0 == (Data.dwFileAttributes & (FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM)) ) {
CSTRING szFilename;
if ( lstrlen(szCurrentDir) > 3 )
szFilename.sprintf(TEXT("%s\\%s\\%s"),szCurrentDir,szDirectory,Data.cFileName);
else
szFilename.sprintf(TEXT("%s%s\\%s"),szCurrentDir,szDirectory,Data.cFileName);
AddMatchFile(ppHead,szFilename);
}
}
}
while ( FindNextFile(hFile,&Data) );
FindClose(hFile);
// Restore old directory
SetCurrentDirectory(szCurrentDir);
}
CSTRING CShimWizard::ShortFile(CSTRING & szStr)
{
LPTSTR szTemp = szStr;
CSTRING szRet;
LPTSTR szWalk = szTemp;
while ( 0 != *szWalk ) {
if ( TEXT('\\') == *szWalk )
szTemp = szWalk+1;
++szWalk;
}
szRet = szTemp;
return szRet;
}
void CShimWizard::GrabMatchingInfo(void)
{
PMATCHENTRY pHead = NULL;
// Delete any matching info which might have already been bound to the record.
PDBENTRY pEntry = m_Record.pEntries;
PDBENTRY pPrev = m_Record.pEntries;
while ( NULL != pEntry ) {
PDBENTRY pHold = pEntry->pNext;
if ( ENTRY_MATCH == pEntry->uType ) {
delete pEntry;
if ( pEntry == m_Record.pEntries ) {
m_Record.pEntries = pHold;
pPrev = m_Record.pEntries;
} else
pPrev->pNext = pHold;
} else
pPrev = pEntry;
pEntry = pHold;
}
// Generate automated matching file information.
WalkDirectory(&pHead,TEXT("."),0);
// Now, take the first X entries and discard the rest.
int nCount = MAX_AUTO_MATCH;
PMATCHENTRY pWalk = pHead;
PMATCHENTRY pTerm = NULL;
while ( NULL != pWalk ) {
if ( 0 >= nCount ) {
PMATCHENTRY pHold = (PMATCHENTRY) pWalk->Entry.pNext;
delete pWalk;
pWalk = pHold;
} else {
--nCount;
if ( 1 == nCount )
pTerm = (PMATCHENTRY) pWalk->Entry.pNext;
pWalk = (PMATCHENTRY) pWalk->Entry.pNext;
}
}
if ( NULL != pTerm )
pTerm->Entry.pNext = NULL;
pWalk = pHead;
while ( NULL != pWalk ) {
GetFileAttributes(pWalk);
pWalk = (PMATCHENTRY)pWalk->Entry.pNext;
}
// Bind this data to the record.
pWalk = pHead;
// Find the end....
while ( NULL != pWalk && NULL != pWalk->Entry.pNext ) {
PMATCHENTRY pHold = (PMATCHENTRY) pWalk->Entry.pNext;
GetFileAttributes(pWalk);
if ( !InsertMatchingInfo(pWalk) )
delete pWalk;
pWalk = pHold;
}
}
BOOL CShimWizard::InsertMatchingInfo(PMATCHENTRY pNew)
{
PDBENTRY pWalk = m_Record.pEntries;
PDBENTRY pHold;
while ( NULL != pWalk ) {
if ( ENTRY_MATCH == pWalk->uType ) {
PMATCHENTRY pThis = (PMATCHENTRY) pWalk;
if ( pThis->szMatchName == pNew->szMatchName )
return FALSE;
if ( 0 < lstrcmpi(pThis->szMatchName,pNew->szMatchName) )
break;
}
pHold = pWalk;
pWalk = pWalk->pNext;
}
if ( pWalk == m_Record.pEntries ) {
pNew->Entry.pNext = m_Record.pEntries;
m_Record.pEntries = (PDBENTRY) pNew;
} else {
pNew->Entry.pNext = pWalk;
pHold->pNext = (PDBENTRY) pNew;
}
return TRUE;
}
INT_PTR CALLBACK
EditCmdLineDlgProc(
HWND hdlg,
UINT uMsg,
WPARAM wParam,
LPARAM lParam
)
/*++
EditCmdLineDlgProc
Description: Handles messages for the edit control.
--*/
{
int wCode = LOWORD(wParam);
int wNotifyCode = HIWORD(wParam);
switch (uMsg) {
case WM_INITDIALOG:
{
SendMessage(
GetDlgItem(hdlg,IDC_SHIM_CMD_LINE), // handle to destination window
EM_LIMITTEXT, // message to send
(WPARAM) 256, // text length
(LPARAM) 0
);
SHAutoComplete(GetDlgItem(hdlg,IDC_SHIM_CMD_LINE), AUTOCOMPLETE);
PSHIMENTRY pShimEntry;
pShimEntry = (PSHIMENTRY)lParam;
SetWindowLongPtr(hdlg, DWLP_USER, lParam);
SetDlgItemText(hdlg, IDC_SHIM_NAME, pShimEntry->szShimName);
if ( pShimEntry->szCmdLine.Length() > 0 ) {
SetDlgItemText(hdlg, IDC_SHIM_CMD_LINE, pShimEntry->szCmdLine);
}
break;
}
case WM_COMMAND:
switch (wCode) {
case IDOK:
{
PSHIMENTRY pShimEntry;
TCHAR szCmdLine[1024] = _T("");
pShimEntry = ( PSHIMENTRY)GetWindowLongPtr(hdlg, DWLP_USER);
GetDlgItemText(hdlg, IDC_SHIM_CMD_LINE, szCmdLine, sizeof(szCmdLine)/sizeof(TCHAR));
if (*szCmdLine != 0) {
pShimEntry->szCmdLine = szCmdLine;
} else {
pShimEntry->szCmdLine.Release();
}
EndDialog(hdlg, TRUE);
break;
}
case IDCANCEL:
EndDialog(hdlg, FALSE);
break;
default:
return FALSE;
}
break;
default:
return FALSE;
}
return TRUE;
}
void SelectShims_TreeDoubleClicked(HWND hDlg)
{
HWND hTree = GetDlgItem(hDlg,IDC_SHIMLIST);
HTREEITEM hItem;
TVITEM tvi;
hItem = TreeView_GetSelection(hTree);
if (hItem == NULL) return;
tvi.hItem = hItem;
tvi.mask = TVIF_HANDLE | TVIF_PARAM;
TreeView_GetItem(hTree, &tvi);
PSHIMENTRY pEntry = (PSHIMENTRY) tvi.lParam;
///////
if (DialogBoxParam(g_hInstance,
MAKEINTRESOURCE(IDD_CMD_LINE),
hDlg,
EditCmdLineDlgProc,
(LPARAM)pEntry)) {
TCHAR szText[1024];
tvi.mask = TVIF_HANDLE | TVIF_TEXT;
if ( pEntry->szCmdLine.Length() == 0 ) {
tvi.pszText = pEntry->szShimName;
} else {
wsprintf(szText, _T("%s - (%s)"), (LPCTSTR)pEntry->szShimName, (LPCTSTR)pEntry->szCmdLine);
tvi.pszText = szText;
}
TreeView_SetItem(hTree, &tvi);
}
///////
}//void SelectShims_TreeDoubleClicked(HWND hDlg)