출처 : http://www.dreamy.pe.kr/zbxe/?mid=codeclip&category=5904&document_srl=5967
어딘가에서 노트패드를 이용한 덤프 방법이 있길래 살펴보았다가 하나 만들어 두는것도 괜찮다 싶어서 만들어 보았습니다.
기능
1. 노트패드가 실행되지 않았을 경우 노트패드를 실행시킨다.
2. 노트패드가 특정 파일을 열어놓은 상태에서도 출력이 원활하게 이루어지도록 한다.
3. 출력은 항상 제일 하단에 위치하도록 한다.
사용 방법
#include "http://pds5.devpia.com/vc_lec/8000/7241/DumpToNotePAD.h"
DumpToNotePAD("노트패드로 내용 출력중..");
DumpToNotePAD("[%s] %d, %.10lf, %s", sVal1, nVal, fVal, sVal2);
DumpToNotePAD("샘플 : %s", sVal);
되도록 MFC 클래스는 배제한 상태로 짰습니다. CPP의 내용은 다음과 같습니다.
#include "Tlhelp32.h"
static HWND hWndNotePad=NULL;
BOOL CALLBACK EnumWndProcForNotePAD(HWND hwnd, LPARAM lParam)
{
DWORD processID;
::GetWindowThreadProcessId(hwnd, &processID);
if (processID == (DWORD)lParam)
{
//동일한 프로세스 핸들.
//프로세스 핸들이 일정한 상태에서 최상위 부모윈도우를 찾는다.
HWND hParent = hwnd;
do
{
hwnd = hParent;
hParent = ::GetWindow(hParent, GW_OWNER);
::GetWindowThreadProcessId(hParent, &processID);
} while (hParent && processID == (DWORD)lParam);
if (::IsWindowVisible(hwnd))
{
//해당 윈도우가 사용가능한 윈도우인지 검사한다.
hWndNotePad = hwnd;
return FALSE; //작업을 종료한다.
}
}
return TRUE;
}
void DumpToNotePAD(char *fmt, ...)
{
char buffer[4096]={0};
va_list argptr;
va_start(argptr, fmt);
vsprintf(buffer, fmt, argptr);
va_end(argptr);
strcat(buffer, "\r\n");
//-------------------------------------------------------------------
//NotePAD를 찾는다.
for (int T=0; T<2; T++)
{
if (hWndNotePad == NULL || !IsWindow(hWndNotePad))
{
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPALL, 0);
if (hSnapshot != (HANDLE)(-1))
{
//설정된 프로세스명을 가진 모든 프로세스ID를 획득한다.
PROCESSENTRY32 Process;
Process.dwSize = sizeof(PROCESSENTRY32);
if (Process32First(hSnapshot, &Process))
{
do
{
char handleName[512]={0};
strcpy(handleName, Process.szExeFile);
int len = strlen(handleName);
if (len > 1 && handleName[1] == ':') //전체 경로 포함
{
char drive[_MAX_DRIVE]={0};
char dir[_MAX_DIR]={0};
char fname[_MAX_FNAME]={0};
char ext[_MAX_EXT]={0};
_splitpath(handleName, drive, dir, fname, ext);
sprintf(handleName, "%s%s", fname, ext);
}
len = strlen(handleName);
//대문자로 변경
for( int i = 0; i < len; i++ )
*(handleName + i) = toupper(*(handleName+i));
//실행 프로세스 이미지명은 너무 길면 확장자 영역이 잘린다.
//따라서 첫부분이 일치하면 해당 프로세스를 찾은것으로 한다.
if (strstr(handleName, "NOTEPAD"))
::EnumWindows(EnumWndProcForNotePAD, (LPARAM)Process.th32ProcessID);
} while (Process32Next(hSnapshot, &Process));
}
CloseHandle(hSnapshot);
}
}
//NotePAD를 찾지 못한 경우 NotePAD를 실행시킨다.
if (hWndNotePad == NULL || !IsWindow(hWndNotePad))
{
::ShellExecute(NULL, NULL, "NotePAD.exe", NULL, NULL, SW_SHOWNORMAL);
Sleep(100); //NotePAD가 실행될 수 있는 시간을 준다.
}
else
break;
}
if (hWndNotePad == NULL || !IsWindow(hWndNotePad)) return; //예외 처리
//----------------------------------------------------------------------------
//NotePAD의 에디터 창을 찾는다.
//NotePAD는 구성상 첫번째 자식이 에디터 창이다.(SPY++로 확인 가능)
HWND hWnd = ::GetWindow(hWndNotePad, GW_CHILD);
//NotePAD의 하단으로 커서 위치 이동
::SendMessage(hWnd, EM_SETSEL,(DWORD)0,(DWORD)-1);
::SendMessage(hWnd, EM_SETSEL,(DWORD)-1,(DWORD)0);
//내용 출력
::SendMessage(hWnd, EM_REPLACESEL, 0, (LPARAM)buffer);
}
'Programing > Windows' 카테고리의 다른 글
MFC에서 프로세스와 스레드, IPC (0) | 2011.09.14 |
---|---|
[Windows]heap에 관한 화제 (0) | 2009.07.31 |
[Windows]Visual Studio 2005/2008에서 멀티코어 CPU를 활용한 Native C++ 컴파일 (0) | 2009.07.31 |
Visual Studio에서 QT사용하기 (1) | 2009.07.30 |
[MFC]Static Control 배경 투명화 및 글자 겹침 해결 방법 (0) | 2009.07.28 |