|
|
/*++
Copyright (c) 1999-2002 Microsoft Corporation
Module Name:
util.cpp
--*/
#include "precomp.hxx"
#pragma hdrstop
//Current Help Id for Open, Merge, Save and Open project dialog box
WORD g_CurHelpId;
// Number of dialog/message boxes currently open
int g_nBoxCount;
BOOL g_fNoPopups;
HWND MDIGetActive( HWND hwndParent, BOOL *lpbMaximized ) /*++
Routine Description:
Create the command window.
Arguments:
hwndParent - The parent window to the command window. In an MDI document, this is usually the handle to the MDI client window: g_hwndMDIClient
Return Value:
The return value is the handle to the active MDI child window.
NULL if no MDI window has been created.
--*/ { Assert(IsWindow(hwndParent)); return (HWND)SendMessage(hwndParent, WM_MDIGETACTIVE, 0, (LPARAM)lpbMaximized ); }
/*** hGetBoxParent
** ** Synopsis: ** hwnd = hGetBoxParent() ** ** Entry: ** none ** ** Returns: ** ** Description: ** Gets a suitable parent window handle for an ** invocation of a message or dialog box. ** Helper function to util.c functions so declared ** near. ** */
HWND hGetBoxParent() { HWND hCurWnd; int i = 0;
hCurWnd = GetFocus(); if (hCurWnd) { while (GetWindowLong(hCurWnd, GWL_STYLE) & WS_CHILD) { hCurWnd = GetParent(hCurWnd); Dbg((++i < 100)); } } else { hCurWnd = g_hwndFrame; }
return hCurWnd; }
/****************************************************************************
FUNCTION: MsgBox
PURPOSE: General purpose message box routine which takes a pointer to the message text. Provides program title as caption.
****************************************************************************/
int MsgBox( HWND hwndParent, PTSTR szText, UINT wType ) /*++
Routine Description:
Generial purpose message box routine which takes a pointer to a message text and prvoides the program title for the caption of the message box.
Arguments:
hwndParament - Supplies the parent window handle for the message box szText - Supplies a pointer to the message box text. wType - Supplies the message box type (to specify buttons)
Return Value:
Returns the message box return code
--*/
{ int MsgBoxRet = IDOK;
if (g_fNoPopups) { //
// log the string to the command win in case testing
// or when the remote server is running
//
CmdLogFmt (_T("%s\r\n"), szText); } else { // If windbg is minimized we need to restore
// it so that the message box shows up.
if (hwndParent == g_hwndFrame && IsIconic(hwndParent)) { ShowWindow(hwndParent, SW_RESTORE); } g_nBoxCount++; MsgBoxRet = MessageBox(hwndParent, szText, g_MainTitleText, wType); g_nBoxCount--; }
return MsgBoxRet; } /* MsgBox() */
/*** ErrorBox
** ** Returns: ** FALSE ** ** Description: ** Display an error message box with an "Error" title, an OK ** button and a Exclamation Icon. First parameter is a ** reference string in the ressource file. The string ** can contain printf formatting chars, the arguments ** follow from the second parameter onwards. ** */
BOOL ErrorBox( HWND hwnd, UINT type, int wErrorFormat, ... ) { TCHAR szErrorFormat[MAX_MSG_TXT]; TCHAR szErrorText[MAX_VAR_MSG_TXT]; // size is as big as considered necessary
va_list vargs;
// load format string from resource file
Dbg(LoadString(g_hInst, wErrorFormat, (PTSTR)szErrorFormat, MAX_MSG_TXT));
va_start(vargs, wErrorFormat); _vstprintf(szErrorText, szErrorFormat, vargs); va_end(vargs);
if (hwnd == NULL) { hwnd = g_hwndFrame; }
if (type == 0) { type = MB_TASKMODAL; }
MsgBox(g_hwndFrame, (PTSTR)szErrorText, type | MB_OK | MB_ICONINFORMATION); return FALSE; //Keep it always FALSE please
}
/*** InformationBox
** ** Description: ** Display an information message box with an "Information" ** title, an OK button and an Information Icon. ** */
void InformationBox( WORD wDescript ... ) { TCHAR szFormat[MAX_MSG_TXT]; TCHAR szText[MAX_VAR_MSG_TXT]; // size is as big as considered necessary
va_list vargs;
// load format string from resource file
Dbg(LoadString(g_hInst, wDescript, (PTSTR)szFormat, MAX_MSG_TXT));
// set up szText from passed parameters
va_start(vargs, wDescript); _vstprintf(szText, szFormat, vargs); va_end(vargs);
MsgBox(g_hwndFrame, (PTSTR)szText, MB_OK | MB_ICONINFORMATION | MB_TASKMODAL);
return; }
/*** QuestionBox
** ** Synopsis: ** int = QuestionBox(wCaptionId, wMsgFormat, wType, ...) ** ** Entry: ** ** Returns: ** The result of the message box call ** ** Description: ** Display an query box with combination of YES, NO and ** CANCEL buttons and a question mark Icon. ** See ErrorBox for discussion. ** */
int CDECL QuestionBox( WORD wMsgFormat, UINT wType, ... ) { TCHAR szMsgFormat[MAX_MSG_TXT]; TCHAR szMsgText[MAX_VAR_MSG_TXT]; va_list vargs;
//Load format string from resource file
Dbg(LoadString(g_hInst, wMsgFormat, (PTSTR)szMsgFormat, MAX_MSG_TXT));
//Set up szMsgText from passed parameters
va_start(vargs, wType); _vstprintf(szMsgText, szMsgFormat, vargs); va_end(vargs);
return MsgBox(g_hwndFrame, szMsgText, wType | MB_ICONEXCLAMATION | MB_TASKMODAL); } /* QuestionBox() */
/****************************************************************************
FUNCTION: QuestionBox2
PURPOSE: Display an query box with combination of YES, NO and CANCEL buttons and a question mark Icon. The type and the parent window are adjustable.
RETURNS: MessageBox result
****************************************************************************/ int CDECL QuestionBox2( HWND hwnd, WORD wMsgFormat, UINT wType, ... ) { TCHAR szMsgFormat[MAX_MSG_TXT]; TCHAR szMsgText[MAX_VAR_MSG_TXT]; va_list vargs;
//Load format string from resource file
Dbg(LoadString(g_hInst, wMsgFormat, (PTSTR)szMsgFormat, MAX_MSG_TXT));
//Set up szMsgText from passed parameters
va_start(vargs, wType); _vstprintf(szMsgText, szMsgFormat, vargs); va_end(vargs);
return MsgBox(hwnd, szMsgText, wType | MB_ICONEXCLAMATION); } /* QuestionBox2() */
/*** ShowAssert
** ** Synopsis: ** void = ShowAssert(szCond, iLine, szFile) ** ** Entry: ** szCond - tokenized form of the failed condition ** iLine - Line number for the assertion ** szFile - File for the assertion ** ** Returns: ** void ** ** Description: ** Prepare and display a Message Box with szCondition, iLine and ** szFile as fields. ** */ void ShowAssert( PTSTR condition, UINT line, PTSTR file ) { TCHAR text[MAX_VAR_MSG_TXT];
//Build line, show assertion and exit program
_stprintf(text, _T("- Line:%u, File:%Fs, Condition:%Fs"), (WPARAM) line, file, condition);
PTSTR szAssertFile = _T("assert.wbg"); HANDLE hFile = NULL;
hFile = CreateFile(szAssertFile, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL ); if (INVALID_HANDLE_VALUE != hFile) { // Write the text out to the file
DWORD dwBytesWritten = 0; Assert(WriteFile(hFile, text, _tcslen(text), &dwBytesWritten, NULL)); Assert(_tcslen(text) == dwBytesWritten); CloseHandle(hFile); }
int Action = MessageBox(GetDesktopWindow(), text, "Assertion Failure", MB_ABORTRETRYIGNORE); if (Action == IDABORT) { exit(3); } else if (Action == IDRETRY) { DebugBreak(); } } // ShowAssert()
/*** StartDialog
** ** Synopsis: ** int = StartDialog(rcDlgNb, dlgProc, lParam) ** ** Entry: ** rcDlgNb - Resource number of dialog to be openned ** dlgProc - Filter procedure for the dialog ** lParam - Data passed into the dlg proc via LPARAM ** ** Returns: ** Result of the dialog box call ** ** Description: ** Loads and execute the dialog box 'rcDlgNb' (resource ** file string number) associated with the dialog ** function 'dlgProc' ** */
int StartDialog( int rcDlgNb, DLGPROC dlgProc, LPARAM lParam ) { LRESULT result;
//
//Execute Dialog Box
//
g_nBoxCount++; result = DialogBoxParam(g_hInst, MAKEINTRESOURCE(rcDlgNb), hGetBoxParent(), dlgProc, lParam ); Assert(result != (LRESULT)-1); g_nBoxCount--;
return (int)result; }
void ProcessNonDlgMessage(LPMSG Msg) { #if 0
{ DebugPrint("NonDlg msg %X for %p, args %X %X\n", Msg->message, Msg->hwnd, Msg->wParam, Msg->lParam); } #endif
// If a keyboard message is for the MDI , let the MDI client
// take care of it. Otherwise, check to see if it's a normal
// accelerator key (like F3 = find next). Otherwise, just handle
// the message as usual.
if (!TranslateMDISysAccel(g_hwndMDIClient, Msg) && !TranslateAccelerator(g_hwndFrame, g_hMainAccTable, Msg)) { //
// If this is a right-button-down over a child window,
// automatically activate the window's contex menu.
//
if (Msg->message == WM_RBUTTONDOWN && IsChild(g_hwndMDIClient, Msg->hwnd)) { HMENU Menu; PCOMMONWIN_DATA CmnWin; POINT ScreenPt; POINT Pt = {LOWORD(Msg->lParam), HIWORD(Msg->lParam)}; ClientToScreen(Msg->hwnd, &Pt); ScreenPt = Pt; ScreenToClient(g_hwndMDIClient, &Pt); HWND Win = ChildWindowFromPointEx(g_hwndMDIClient, Pt, CWP_SKIPINVISIBLE); if (Win != NULL && (CmnWin = GetCommonWinData(Win)) != NULL && (Menu = CmnWin->GetContextMenu()) != NULL) { UINT Item = TrackPopupMenu(Menu, TPM_LEFTALIGN | TPM_TOPALIGN | TPM_NONOTIFY | TPM_RETURNCMD | TPM_RIGHTBUTTON, ScreenPt.x, ScreenPt.y, 0, Msg->hwnd, NULL); if (Item) { CmnWin->OnContextMenuSelection(Item); } return; } } TranslateMessage(Msg); DispatchMessage(Msg); } }
void ProcessPendingMessages(void) { MSG Msg; // Process all available messages.
while (PeekMessage(&Msg, NULL, 0, 0, PM_NOREMOVE)) { if (!GetMessage(&Msg, NULL, 0, 0)) { g_Exit = TRUE; break; }
if (g_FindDialog == NULL || !IsDialogMessage(g_FindDialog, &Msg)) { ProcessNonDlgMessage(&Msg); } } }
/****************************************************************************
FUNCTION: InfoBox
PURPOSE: Opens a Dialog box with a title and accepting a printf style for text. It's for DEBUGGING USE ONLY
****************************************************************************/ int InfoBox( PTSTR text, ... ) { TCHAR buffer[MAX_MSG_TXT]; va_list vargs;
va_start(vargs, text); _vstprintf(buffer, text, vargs); va_end(vargs); return MsgBox(GetActiveWindow(), buffer, MB_OK | MB_ICONINFORMATION | MB_TASKMODAL); }
void ExitDebugger(PDEBUG_CLIENT Client, ULONG Code) { if (Client != NULL) { if (g_RemoteClient) { // Disconnect from server.
Client->EndSession(DEBUG_END_DISCONNECT); } else { Client->EndSession(DEBUG_END_PASSIVE); // Force servers to get cleaned up.
Client->EndSession(DEBUG_END_REENTRANT); } }
ExitProcess(Code); }
// XXX drewb - Is this functionality present in other utilities?
// FatalErrorBox is close. Probably can combine something here.
void ErrorExit(PDEBUG_CLIENT Client, PCSTR Format, ...) { char Message[1024]; va_list Args;
va_start(Args, Format); _vsnprintf(Message, sizeof(Message), Format, Args); va_end(Args);
// XXX drewb - Could put up message box.
OutputDebugString(Message);
ExitDebugger(Client, E_FAIL); }
#define MAX_FORMAT_STRINGS 8
LPSTR FormatAddr64( ULONG64 addr ) /*++
Routine Description:
Format a 64 bit address, showing the high bits or not according to various flags.
An array of static string buffers is used, returning a different buffer for each successive call so that it may be used multiple times in the same printf.
Arguments:
addr - Supplies the value to format
Return Value:
A pointer to the string buffer containing the formatted number
--*/ { static CHAR strings[MAX_FORMAT_STRINGS][20]; static int next = 0; LPSTR string;
string = strings[next]; ++next; if (next >= MAX_FORMAT_STRINGS) { next = 0; } if (g_Ptr64) { sprintf(string, "%08x`%08x", (ULONG)(addr>>32), (ULONG)addr); } else { sprintf(string, "%08x", (ULONG)addr); } return string; }
static BOOL FAddToSearchPath = FALSE; static BOOL FAddToRootMap = FALSE;
/*
** AppendFilter ** ** Description: ** Append a filter to an existing filters string. ** */
BOOL AppendFilter( WORD filterTextId, int filterExtId, PTSTR filterString, int *len, int maxLen ) { int size; TCHAR Tmp[MAX_MSG_TXT];
//
// Append filter text
//
Dbg(LoadString(g_hInst, filterTextId, Tmp, MAX_MSG_TXT)); size = _tcslen(Tmp) + 1; if (*len + size > maxLen) { return FALSE; } memmove(filterString + *len, Tmp, size); *len += size;
//
// Append filter extension
//
Dbg(LoadString(g_hInst, filterExtId, Tmp, MAX_MSG_TXT)); size = _tcslen(Tmp) + 1; if (*len + size > maxLen) { return FALSE; } memmove(filterString + *len, Tmp, size); *len += size;
return TRUE; }
/*
** InitFilterString ** ** Description: ** Initialize file filters for file dialog boxes. */
void InitFilterString( WORD titleId, PTSTR filter, int maxLen ) { int len = 0;
switch (titleId) { case DLG_Browse_CrashDump_Title: AppendFilter(TYP_File_DUMP, DEF_Ext_DUMP, filter, &len, maxLen); break;
case DLG_Browse_Executable_Title: AppendFilter(TYP_File_EXE, DEF_Ext_EXE, filter, &len, maxLen); break;
case DLG_Browse_LogFile_Title: AppendFilter(TYP_File_LOG, DEF_Ext_LOG, filter, &len, maxLen); break;
case DLG_Open_Filebox_Title: case DLG_Browse_Filebox_Title: case DLG_Browse_Source_Path_Title: AppendFilter(TYP_File_SOURCE, DEF_Ext_SOURCE, filter, &len, maxLen); AppendFilter(TYP_File_INCLUDE, DEF_Ext_INCLUDE, filter, &len, maxLen); AppendFilter(TYP_File_ASMSRC, DEF_Ext_ASMSRC, filter, &len, maxLen); AppendFilter(TYP_File_INC, DEF_Ext_INC, filter, &len, maxLen); AppendFilter(TYP_File_RC, DEF_Ext_RC, filter, &len, maxLen); AppendFilter(TYP_File_DLG, DEF_Ext_DLG, filter, &len, maxLen); AppendFilter(TYP_File_DEF, DEF_Ext_DEF, filter, &len, maxLen); AppendFilter(TYP_File_MAK, DEF_Ext_MAK, filter, &len, maxLen); break;
case DLG_Browse_DbugDll_Title: AppendFilter(TYP_File_DLL, DEF_Ext_DLL, filter, &len, maxLen); break;
case DLG_Browse_Symbol_Path_Title: AppendFilter(TYP_File_Symbols, DEF_Ext_Symbols, filter, &len, maxLen); AppendFilter(TYP_File_EXE, DEF_Ext_EXE, filter, &len, maxLen); AppendFilter(TYP_File_DLL, DEF_Ext_DLL, filter, &len, maxLen); break; case DLG_Browse_Image_Path_Title: AppendFilter(TYP_File_EXE, DEF_Ext_EXE, filter, &len, maxLen); AppendFilter(TYP_File_DLL, DEF_Ext_DLL, filter, &len, maxLen); break; case DLG_Write_Text_File_Title: AppendFilter(TYP_File_TXT, DEF_Ext_TXT, filter, &len, maxLen); break;
case DLG_Open_Workspace_File_Title: case DLG_Save_Workspace_File_Title: AppendFilter(TYP_File_Workspace, DEF_Ext_Workspace, filter, &len, maxLen); break;
default: Assert(FALSE); break; }
AppendFilter(TYP_File_ALL, DEF_Ext_ALL, filter, &len, maxLen); filter[len] = _T('\0'); }
BOOL StartFileDlg( HWND hwnd, int titleId, int defExtId, int helpId, int templateId, PTSTR InitialDir, PTSTR fileName, DWORD* pFlags, LPOFNHOOKPROC lpfnHook )
/*++
Routine Description:
This function is used by windbg to open the set of common file handling dialog boxes.
Arguments:
hwnd - Supplies the wnd to hook the dialog box to
titleId - Supplies the string resource of the title
defExtId - Supplies The default extension resource string
helpId - Supplies the help number for the dialog box
templateId - Supplies the dialog resource number if non-zero
fileName - Supplies the default file name
pFiles - Supplies a pointer to flags
lpfnHook - Supplies the address of a hook procedure for the dialog
Return Value:
The result of the dialog box call (usually TRUE for OK and FALSE for cancel)
--*/
{ #define filtersMaxSize 350
OPENFILENAME_NT4 OpenFileName = {0}; TCHAR title[MAX_MSG_TXT]; TCHAR defExt[MAX_MSG_TXT]; BOOL result; TCHAR filters[filtersMaxSize]; LPOFNHOOKPROC lpDlgHook = NULL; HCURSOR hSaveCursor; TCHAR files[_MAX_PATH + 8]; TCHAR szExt[_MAX_EXT + 8]; TCHAR szBase[_MAX_PATH + 8]; int indx; TCHAR fname[_MAX_FNAME]; TCHAR ext[_MAX_EXT]; PTSTR LocalInitialDir = NULL;
*pFlags |= (OFN_EXPLORER | OFN_NOCHANGEDIR);
if (InitialDir == NULL || !InitialDir[0]) { DWORD retval = GetCurrentDirectory(NULL, NULL); InitialDir = (PTSTR)calloc(retval, sizeof(TCHAR) ); if (InitialDir == NULL) { return FALSE; } GetCurrentDirectory(retval, InitialDir); LocalInitialDir = InitialDir; }
if (DLG_Browse_Filebox_Title == titleId) { _tsplitpath( fileName, NULL, NULL, fname, ext ); _tmakepath( files, NULL, NULL, fname, ext ); } else { _tcscpy(files, fileName); }
//
// Set the Hour glass cursor
//
hSaveCursor = SetCursor(LoadCursor(NULL, IDC_WAIT));
InitFilterString((WORD)titleId, (PTSTR)filters, (int)filtersMaxSize); Dbg(LoadString(g_hInst, titleId, (PTSTR)title, MAX_MSG_TXT)); Dbg(LoadString(g_hInst, defExtId, (PTSTR)defExt, MAX_MSG_TXT)); if (templateId) { //
// Build dialog box Name
//
*pFlags |= OFN_ENABLETEMPLATE; OpenFileName.lpTemplateName = MAKEINTRESOURCE(templateId); } else { *pFlags |= OFN_EXPLORER; }
//
// Make instance for _T('dlgProc')
//
if (lpfnHook) { lpDlgHook = lpfnHook;
*pFlags |= OFN_ENABLEHOOK; }
g_CurHelpId = (WORD) helpId; OpenFileName.lStructSize = sizeof(OpenFileName); OpenFileName.hwndOwner = hwnd; OpenFileName.hInstance = g_hInst; OpenFileName.lpstrFilter = (PTSTR)filters; OpenFileName.lpstrCustomFilter = NULL; OpenFileName.nMaxCustFilter = 0; OpenFileName.nFilterIndex = 1; OpenFileName.lpstrFile = files; OpenFileName.nMaxFile = _MAX_PATH; OpenFileName.lpstrFileTitle = NULL; OpenFileName.lpstrInitialDir = InitialDir; OpenFileName.lpstrTitle = (PTSTR)title; OpenFileName.Flags = *pFlags; OpenFileName.lpstrDefExt = (PTSTR)NULL; OpenFileName.lCustData = 0L; OpenFileName.lpfnHook = lpDlgHook;
g_nBoxCount++;
switch (titleId) { case DLG_Open_Filebox_Title: _tcscat(OpenFileName.lpstrFile, defExt); // fall thru
case DLG_Browse_Executable_Title: case DLG_Browse_CrashDump_Title: case DLG_Browse_Symbol_Path_Title: case DLG_Browse_Source_Path_Title: case DLG_Browse_Image_Path_Title: case DLG_Open_Workspace_File_Title: result = GetOpenFileName((LPOPENFILENAME)&OpenFileName); break;
case DLG_Browse_LogFile_Title: if (fileName) { _tcscpy(OpenFileName.lpstrFile, fileName); } else { *OpenFileName.lpstrFile = 0; } result = GetOpenFileName((LPOPENFILENAME)&OpenFileName); break;
case DLG_Browse_DbugDll_Title: *(OpenFileName.lpstrFile) = _T('\0'); result = GetOpenFileName((LPOPENFILENAME)&OpenFileName); break;
case DLG_Browse_Filebox_Title: _tsplitpath (files, (PTSTR)NULL, (PTSTR)NULL, (PTSTR)szBase, szExt); indx = matchExt (szExt, filters);
if (indx != -1) { OpenFileName.nFilterIndex = indx; }
_tcscat(title, szBase); if (*szExt) { _tcscat(title, szExt); }
FAddToSearchPath = FALSE; FAddToRootMap = FALSE;
result = GetOpenFileName((LPOPENFILENAME)&OpenFileName);
//
// Check to see if the use said to add a file to the browse path.
// If so then add it to the front of the path
//
/*if (FAddToSearchPath)
{ AddToSearchPath(OpenFileName.lpstrFile); } else if (FAddToRootMap) { RootSetMapped(fileName, OpenFileName.lpstrFile); }*/ break;
case DLG_Write_Text_File_Title: case DLG_Save_Workspace_File_Title: OpenFileName.lpstrDefExt = defExt; *(OpenFileName.lpstrFile) = _T('\0'); _tcscat(OpenFileName.lpstrFile, defExt); result = GetSaveFileName((LPOPENFILENAME)&OpenFileName); break; default: Assert(FALSE); free(LocalInitialDir); return FALSE; }
g_nBoxCount--;
if (result && titleId == DLG_Open_Filebox_Title) { KNOWN_EXT ExtType = RecognizeExtension(OpenFileName.lpstrFile); if (ExtType != KNOWN_EXT_COUNT) { if (QuestionBox(STR_May_Not_Be_Source, MB_YESNO) != IDYES) { result = 0; } } } if (result) { _tcscpy(fileName, OpenFileName.lpstrFile); switch(titleId) { case DLG_Open_Filebox_Title: AddFileToMru(FILE_USE_SOURCE, fileName); break; case DLG_Browse_Symbol_Path_Title: case DLG_Browse_Source_Path_Title: case DLG_Browse_Image_Path_Title: // Return only the path.
if (OpenFileName.nFileOffset > 0) { fileName[OpenFileName.nFileOffset - 1] = 0; } break; }
//
// Get the output of flags
//
*pFlags = OpenFileName.Flags; }
//
//Restore cursor
//
SetCursor(hSaveCursor);
free(LocalInitialDir); return result; } /* StartFileDlg() */
/*** matchExt
** ** Synopsis: ** int = matchExt (queryExtension, sourceList) ** ** Entry: ** ** Returns: 1-based index of pairwise substring for which the second ** element (i.e., the extension list), contains the target ** extension. If there is no match, we return -1. ** ** Description: ** Searches extension lists for the Open/Save/Browse common ** dialogs to try to match a filter to the input filename's ** extension. ** (Open File, Save File, Merge File and Open Project) ** ** Implementation note: Our thinking looks like this: ** ** We are given a sequence of null-terminated strings which ** are text/extension pairs. We return the pairwise 1-based ** index of the first pair for which the second element has an ** exact match for the target extension. (Everything, by the ** way, is compared without case sensitivity.) We picture the ** source sequence, then, to be an array whose elements are pairs ** of strings (we will call the pairs 'left' and 'right'). ** ** Just to complicate things, we allow the '.right' pair elements to ** be strings like "*.c;*.cpp;*.cxx", where we our query might be ** any one of the three (minus the leading asterisk). Fortunately, ** _tcstok() will break things apart for us (see the 'delims[]' array ** in the code for the delimiters we have chosen). ** ** Assuming there is a match in there somewhere, our invariant ** for locating the first one will be: ** ** Exists(k): ** ForAll(i) : 0 <= i < k ** : queryExtension \not IS_IN source[i].right ** \and ** queryExtension IS_IN source[k].right ** ** where we define IS_IN to be a membership predicate (using _tcstok() ** and _tcsicmp() in the implementation, eh?): ** ** x IS_IN y ** <=> ** Exists (t:token) : (t \in y) \and (x == t). ** ** The guard for our main loop, then, comes from the search for the ** queryExtension within the tokens inside successive '.right' elements. ** We choose to continue as long as there is no current token in the ** pair's right side that contains the query. ** ** (We have the pragmatic concern that the value may not be there, so we ** augment the loop guard with the condition that we have not yet ** exhausted the source. This is straightforward to add to the ** invariant, but it causes a lot of clutter that does help our ** comprehension at all, so we just stick it in the guard without ** formal justification.) */
int matchExt( PTSTR queryExtension, PTSTR sourceList ) { int answer; int idxPair = 1; // a 1-based index!
PTSTR tokenMatch = 0;
TCHAR delims[] = _T("*,; ") ; // Given a typical string: "*.c;*.cpp;*.cxx",
// _tcstok() would produce three tokens:
// ".c", ".cpp", and ".cxx".
while (*sourceList != 0 && tokenMatch == 0) { while (*sourceList != _T('\0')) { sourceList++; } // skip first string of pair
sourceList++; // and increment beyond NULL
if (*sourceList != _T('\0')) { PTSTR work = _tcsdup (sourceList); // copy to poke holes in
tokenMatch = _tcstok (work, delims);
while (tokenMatch && _tcsicmp (tokenMatch, queryExtension)) { tokenMatch = _tcstok (0, delims); }
free (work); }
if (tokenMatch == 0) // no match: need to move to next pair
{ while (*sourceList != _T('\0')) { sourceList++; } // skip second string of pair
sourceList++; // and increment beyond NULL
idxPair++; } }
answer = (tokenMatch != 0) ? idxPair : (-1);
return (answer); }
/*** DlgFile
** ** Synopsis: ** bool = DlgFile(hDlg, message, wParam, lParam) ** ** Entry: ** ** Returns: ** ** Description: ** Processes messages for file dialog boxes ** Those dialogs are not called directly but are called ** by the DlgFile function which contains all basic ** elements for Dialogs Files Operations Handling. ** (Open File, Save File, Merge File and Open Project) ** ** See OFNHookProc */
UINT_PTR APIENTRY DlgFile( HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam )
{ switch (uMsg) { case WM_NOTIFY: { LPOFNOTIFY lpon = (LPOFNOTIFY) lParam;
//
// Determine what happened/why we are being notified
//
switch (lpon->hdr.code) { case CDN_HELP: // Help button pushed
Dbg(HtmlHelp(hDlg,g_HelpFileName, HH_HELP_CONTEXT, g_CurHelpId)); break; } } break; } return FALSE; } /* DlgFile() */
UINT_PTR APIENTRY GetOpenFileNameHookProc( HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam )
/*++
Routine Description:
This routine is handle the Add Directory To radio buttons in the browser source file dialog box.
Arguments:
hDlg - Supplies the handle to current dialog msg - Supplies the message to be processed wParam - Supplies info about the message lParam - Supplies info about the message
Return Value:
TRUE if we replaced default processing of the message, FALSE otherwise
--*/ { /*
switch( msg ) { case WM_INITDIALOG: return TRUE;
case WM_NOTIFY: { LPOFNOTIFY lpon = (LPOFNOTIFY) lParam;
switch(lpon->hdr.code) { case CDN_FILEOK: FAddToSearchPath = (IsDlgButtonChecked( hDlg, IDC_CHECK_ADD_SRC_ROOT_MAPPING) == BST_CHECKED); return 0; } } } return DlgFile(hDlg, msg, wParam, lParam); */ return 0; } /* GetOpenFileNameHookProc() */
void Internal_Activate( HWND hwndPrev, HWND hwndCur, HWND hwndNew, int nPosition ) /*++
Routine Description:
Places a window in the specified Z order position.
Arguments:
hwndPrev - Window prior to hwndCur. Can be NULL. hwndCur - Currently active window, topmost in Z order. Can be NULL.
hwndNew - The window to be placed in the new Z order.
nPosition - Where the window is to be place in the Z order. 1 - topmost 2 - 2nd place (behind topmost) 3 - 3rd place, etc....
Return Value:
None
--*/ { // Sanity check. Make sure the programmer
// specified a 1, 2, or 3. We are strict in order to
// keep it readable.
Assert(1 <= nPosition && nPosition <= 3); Assert(hwndNew);
switch (nPosition) { case 1: // Make it topmost
SendMessage(g_hwndMDIClient, WM_MDIACTIVATE, (WPARAM) hwndNew, 0); break;
case 2: // Try to place it 2nd in Z order
if (NULL == hwndCur) { // We don't have a topmost window,
// so make this one the topmost window
SendMessage(g_hwndMDIClient, WM_MDIACTIVATE, (WPARAM) hwndNew, 0); } else { // Place it in 2nd
SetWindowPos(hwndNew, hwndCur, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
// Give the topmost most focus again and activate UI visual clues.
SendMessage(g_hwndMDIClient, WM_MDIACTIVATE, (WPARAM) hwndCur, 0); } break;
case 3: // Try to place it 3rd in Z order
if (NULL == hwndCur) { // We don't have a topmost window,
// so make this one the topmost window
SendMessage(g_hwndMDIClient, WM_MDIACTIVATE, (WPARAM) hwndNew, 0); } else { // Is there a window 2nd in the Z order?
if (NULL == hwndPrev) { // No window 2nd in Z order. Then simply place it after the
// topmost window.
hwndPrev = hwndCur; }
// Place it
SetWindowPos(hwndNew, hwndPrev, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
// Give the topmost most focus again and activate UI visual clues.
SendMessage(g_hwndMDIClient, WM_MDIACTIVATE, (WPARAM) hwndCur, 0); } break;
default: // Sanity check, the programmer missed a case
Assert(0); } }
void ReorderChildren( HWND hwndPrev, HWND hwndCur, HWND hwndNew, BOOL bUserActivated ) { PCOMMONWIN_DATA pCur_WinData = NULL; PCOMMONWIN_DATA pNew_WinData = NULL; PCOMMONWIN_DATA pPrev_WinData = NULL;
if (hwndCur) { pCur_WinData = GetCommonWinData(hwndCur); }
pNew_WinData = GetCommonWinData(hwndNew); Assert(pNew_WinData); if (!pNew_WinData) { return; }
if (hwndPrev) { pPrev_WinData = GetCommonWinData(hwndPrev); }
//
// Handle the case where the window activation
// was requested by the debugger itself and not the
// user.
//
switch (pNew_WinData->m_enumType) { default: Internal_Activate(hwndPrev, hwndCur, hwndNew, bUserActivated ? 2 : 1); break;
case DISASM_WINDOW: case DOC_WINDOW: if (GetSrcMode_StatusBar()) { // Src mode
if (pCur_WinData != NULL && (DISASM_WINDOW == pCur_WinData->m_enumType || DOC_WINDOW == pCur_WinData->m_enumType)) { // We can take the place of another doc/asm wind
// Place 1st in z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 1); } else { if (pPrev_WinData != NULL && (DOC_WINDOW == pPrev_WinData->m_enumType || DISASM_WINDOW == pPrev_WinData->m_enumType)) { // Don't have a window in 2nd place, or if we do it
// is a src or asm window, and we can hide it.
// Place 2nd in Z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 2); } else { // Place 3rd in Z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 3); } } } else { WIN_TYPES Type = pCur_WinData != NULL ? pCur_WinData->m_enumType : MINVAL_WINDOW; // Asm mode
// Which is currently the topmost window.
switch (Type) { case DOC_WINDOW: // Place 1st in z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 1); break;
case DISASM_WINDOW: if (DOC_WINDOW == pNew_WinData->m_enumType) { if (pPrev_WinData == NULL || DOC_WINDOW != pPrev_WinData->m_enumType) { // We have a window in second place that isn't a doc
// window (locals, watch, ...).
Internal_Activate(hwndPrev, hwndCur, hwndNew, 3); } else { // Either don't have any windows in second place, or
// we have a window in second place that is a doc
// window. We can take its place.
//
// Place 2nd in z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 2); } } else { // Should never happen. The case of disasm being activated
// when it is currently active should ahve already been
// taken care of.
Dbg(0); } break;
default: if ((pPrev_WinData != NULL && DISASM_WINDOW == pPrev_WinData->m_enumType) && DOC_WINDOW == pNew_WinData->m_enumType) { // window (locals, watch, ...).
Internal_Activate(hwndPrev, hwndCur, hwndNew, 3); } else { // Place 2nd in z-order
Internal_Activate(hwndPrev, hwndCur, hwndNew, 2); } break; } } break; } }
void ActivateMDIChild( HWND hwndNew, BOOL bUserActivated ) /*++
Routine Description: Used to activate a specified window. Automatically uses the hwndActive variable to determine the currently active window.
Arguments: hwndNew - The window to be placed in the new Z order. bUserActivated - Indicates whether this action was initiated by the user or by windbg. The value is to determine the Z order of any windows that are opened. --*/ { if (hwndNew == NULL) { Assert(hwndNew); return; }
HWND hwndPrev = NULL;
HWND hwndCur = MDIGetActive(g_hwndMDIClient, NULL);
if (!hwndCur || bUserActivated || hwndCur == hwndNew) { // Nothing else was open. So we make this one the
// topmost window.
//
// Or the user requested that this window be made
// the topmost window, and we obey.
//
// Or we are re-activating the current window.
Internal_Activate(NULL, hwndCur, hwndNew, 1); return; }
// See is we have 3 or more windows open
hwndPrev = GetNextWindow(hwndCur, GW_HWNDNEXT);
ReorderChildren(hwndPrev, hwndCur, hwndNew, bUserActivated); }
void AppendTextToAnEditControl( HWND hwnd, PTSTR pszNewText ) { Assert(hwnd); Assert(pszNewText);
CHARRANGE chrrgCurrent = {0}; CHARRANGE chrrgAppend = {0};
// Get the current selection
SendMessage(hwnd, EM_EXGETSEL, 0, (LPARAM) &chrrgCurrent);
// Set the selection to the very end of the edit control
chrrgAppend.cpMin = chrrgAppend.cpMax = GetWindowTextLength(hwnd); SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM) &chrrgCurrent); // Append the text
SendMessage(hwnd, EM_REPLACESEL, FALSE, (LPARAM) pszNewText);
// Restore previous selection
SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM) &chrrgCurrent); }
VOID CopyToClipboard( PSTR str, BOOL ExpandLf ) { if (!str) { return; }
ULONG Len = strlen(str) + 1;
if (ExpandLf) { // Rather than bother to count the actual number
// of linefeeds that need to be expanded, just
// allocate twice as much memory as necessary. There's
// only one thing on the clipboard at a time and
// the things copied here are relatively small so this
// isn't a problem.
Len *= 2; } HANDLE Mem = GlobalAlloc(GMEM_MOVEABLE, Len); if (Mem == NULL) { return; }
PSTR Text = (PSTR)GlobalLock(Mem); if (Text == NULL) { GlobalFree(Mem); return; }
if (ExpandLf) { PSTR Rd, Wr;
Rd = str; Wr = Text; while (*Rd) { if (*Rd == '\n') { *Wr++ = '\r'; } *Wr++ = *Rd++; } *Wr = 0; } else { strcpy(Text, str); }
GlobalUnlock(Mem);
if (OpenClipboard(NULL)) { EmptyClipboard(); if (SetClipboardData(CF_TEXT, Mem) == NULL) { GlobalFree(Mem); }
CloseClipboard(); } }
void SetAllocString(PSTR* Str, PSTR New) { if (*Str != NULL) { free(*Str); } *Str = New; }
BOOL DupAllocString(PSTR* Str, PSTR New) { PSTR NewStr = (PSTR)malloc(strlen(New) + 1); if (NewStr == NULL) { return FALSE; }
strcpy(NewStr, New); SetAllocString(Str, NewStr); return TRUE; }
BOOL PrintAllocString(PSTR* Str, int Len, PCSTR Format, ...) { PSTR NewStr = (PSTR)malloc(Len); if (NewStr == NULL) { return FALSE; } va_list Args;
va_start(Args, Format); if (_vsnprintf(NewStr, Len, Format, Args) < 1) { NewStr[Len - 1] = 0; } va_end(Args);
SetAllocString(Str, NewStr); return TRUE; }
HMENU CreateContextMenuFromToolbarButtons(ULONG NumButtons, TBBUTTON* Buttons, ULONG IdBias) { ULONG i; HMENU Menu;
Menu = CreatePopupMenu(); if (Menu == NULL) { return Menu; }
for (i = 0; i < NumButtons; i++) { MENUITEMINFO Item;
ZeroMemory(&Item, sizeof(Item)); Item.cbSize = sizeof(Item); Item.fMask = MIIM_TYPE; if (Buttons->fsStyle & BTNS_SEP) { Item.fType = MFT_SEPARATOR; } else { Item.fMask |= MIIM_ID; Item.fType = MFT_STRING; Item.wID = (WORD)(Buttons->idCommand + IdBias); Item.dwTypeData = (LPSTR)Buttons->iString; } if (!InsertMenuItem(Menu, i, TRUE, &Item)) { DestroyMenu(Menu); return NULL; }
Buttons++; }
DrawMenuBar(g_hwndFrame); return Menu; }
HWND AddButtonBand(HWND Bar, PTSTR Text, PTSTR SizingText, UINT Id) { HWND Button; HDC Dc; RECT Rect; Button = CreateWindowEx(0, "BUTTON", Text, WS_VISIBLE | WS_CHILD, 0, 0, 0, 0, Bar, (HMENU)(UINT_PTR)Id, g_hInst, NULL); if (Button == NULL) { return NULL; }
Rect.left = 0; Rect.top = 0; SendMessage(Button, WM_SETFONT, (WPARAM)g_Fonts[FONT_VARIABLE].Font, 0); Dc = GetDC(Button); if (Dc != NULL) { SIZE Size; GetTextExtentPoint32(Dc, SizingText, strlen(SizingText), &Size); Rect.right = Size.cx; Rect.bottom = Size.cy; ReleaseDC(Button, Dc); } else { Rect.right = strlen(Text) * g_Fonts[FONT_FIXED].Metrics.tmAveCharWidth; Rect.bottom = g_Fonts[FONT_FIXED].Metrics.tmHeight; }
REBARBANDINFO BandInfo; BandInfo.cbSize = sizeof(BandInfo); BandInfo.fMask = RBBIM_STYLE | RBBIM_CHILD | RBBIM_CHILDSIZE; BandInfo.fStyle = RBBS_FIXEDSIZE; BandInfo.hwndChild = Button; BandInfo.cxMinChild = Rect.right - Rect.left + 4 * GetSystemMetrics(SM_CXEDGE); BandInfo.cyMinChild = Rect.bottom - Rect.top + 2 * GetSystemMetrics(SM_CYEDGE); SendMessage(Bar, RB_INSERTBAND, -1, (LPARAM)&BandInfo);
return Button; }
KNOWN_EXT RecognizeExtension(PTSTR Path) { PSTR Scan;
//
// Find the extension.
//
Scan = Path + strlen(Path); for (;;) { if (Scan == Path) { return KNOWN_EXT_COUNT; }
if (*--Scan == '.') { Scan++; break; } }
if (!_stricmp(Scan, "dmp") || !_stricmp(Scan, "mdmp")) { return EXT_DUMP; } else if (!_stricmp(Scan, "exe")) { return EXT_EXE; }
return KNOWN_EXT_COUNT; }
|