블로그 이미지
Kanais
Researcher & Developer 퍼즐을 완성하려면 퍼즐 조각들을 하나 둘씩 맞춰나가야 한다. 인생의 퍼즐 조각들을 하나 둘씩 맞춰나가다 보면 인생이란 퍼즐도 완성되는 날이 오려나...?

calendar

1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31

Notice

2015. 5. 11. 16:10 Programming/Win32 API

작성날짜    : 2011-03-18


출처 : http://blog.naver.com/jcw86/60053729091

PI 프로그래밍에 대한 Q&A입니다.

1. 특정 디렉토리 뒤지기
지정한 디렉토리에 있는 모든 파일을 찾아내는 코드를 만들려면 어떻게 해야 합니까 ?

이 때 사용할 수 있는 API가 바로 FindFirstFile과 FindNextFile, FindClose라는 API들입니다. 사용 예제는 다음과 같습니다.

 WIN32_FIND_DATA  findFileData;
 HANDLE hFileHandle;

 // szDir에 뒤지고자 하는 디렉토리의 경로명을 준다. 예를 들면 "C:\\TEMP\\*.*"
        // 찾아진 파일의 속성은 findFileData의 dwFileAttributes를 살펴본다.
 hFileHandle = FindFirstFile(m_szDir, &findFileData);  
 if (hFileHandle != INVALID_HANDLE_VALUE)   // 파일을 찾은 경우 
 {
  // 찾은 파일의 이름은 cFileName 필드로 들어온다.
  ...
  // 다음 파일을 찾는다.
  while(FindNextFile(hFileHandle, &findFileData))   {
   ...
  }
  FindClose(hFileHandle);
 } 


2. API를 이용하는 유니코드와 ANSI 문자열간의 변환 방법
API를 이용해서 유니코드와 ANSI 문자열간의 변환은 어떻게 수행합니까 ?

Visual C++에서 유니코드 문자열은 BSTR이란 타입으로 표시됩니다. 또 유니코드와 ANSI 문자열간의 변환을 위해서 윈도우 시스템에는 MultiByteToWideChar와 WideCharToMultiByte라는 API가 존재합니다. MFC에서의 BSTR 타입 변환방법이나 ATL로 하는 BSTR 타입 변환도 참고하시기 바랍니다.


ANSI 문자열에서 유니코드로의 변환 방법 
 // sTime이란 ANSI 문자열을 bstr이란 이름의 유니코드(BSTR 타입) 변수로 변환
 char sTime[] = "유니코드 변환 예제";
 BSTR bstr;
 // sTime을 유니코드로 변환하기에 앞서 먼저 그 길이를 알아야 한다.
 int nLen = MultiByteToWideChar(CP_ACP, 0, sTime, lstrlen(sTime), NULL, NULL);
 // 얻어낸 길이만큼 메모리를 할당한다.
 bstr = SysAllocStringLen(NULL, nLen);
 // 이제 변환을 수행한다.
 MultiByteToWideChar(CP_ACP, 0, sTime, lstrlen(sTime), bstr, nLen);
         // 필요없어지면 제거한다.
         SysFreeString(bstr);
유니코드에서 ANSI 문자열로의 변환 방법 
 // newVal이란 BSTR 타입에 있는 유니코드 문자열을 sTime이라는 ANSI 문자열로 변환
 char *sTime;
         int nLen = WideCharToMultiByte(CP_ACP, 0, newVal, -1, sTime, 0, NULL, NULL);
         sTime = malloc(nLen+1);
 WideCharToMultiByte(CP_ACP, 0, newVal, -1, sTime, 128, NULL, NULL);
        // 필요없으면 메모리를 제거한다.
        free(sTime);
유니코드 문자열을 UTF-8으로 변환하기 
     WideCharToMultiByte 함수를 호출할 때 첫 번째 인자로 CP_UTF8을 지정하면 된다. UTF-8은 유니코드의 인코딩 스킴 중의 하나로 쉽게 말하자면 문자열 스트림에서 0을 빼고 표현하는 방법이라고 볼 수 있다.

3. 레지스트리 읽기/쓰기
API를 이용해서 레지스트리에 한 항목을 생성하거나 기존 항목의 값을 읽어들이려면 어떻게 해야합니까 ?

레지스트리 관련 API를 사용하려면 winreg.h라는 헤더 파일을 소스에 포함해야 합니다. 레지스트리에 키를 생성하는 방법과 레지스트리에 존재하는 키의 값을 읽는 방법을 차례로 살펴보겠습니다.


레지스트리 키 생성 예제 
 // 예를 들어 HKEY_LOCAL_MACHINE밑의 System\CurrentControlSet\Services\GenPort라는 키를
        // 생성하고 거기에 DWORD 타입의 값으로 Type을 만들고 문자열 타입의 값으로 Group
        // 을 만들어 본다.
 #include "winreg.h"
 LONG error = 0;
 HKEY hKey;
 DWORD dwDisp, dwData;
 char lpData[] = "Write this down";

 // 먼저 만들려는 키가 이미 존재하는 것인지 살혀본다.
 error = RegOpenKeyEx(HKEY_LOCAL_MACHINE, "System\\CurrentControlSet\\Services\\GenPort",
                         0, KEY_ALL_ACCESS, &hKey);

 if (error != ERROR_SUCCESS) // 없다면 새로 생성한다.
 {
  // 키를 생성한다.
  error = RegCreateKeyEx(HKEY_LOCAL_MACHINE,
   "System\\CurrentControlSet\\Services\\GenPort", 0, "REG_BINARY", 
          REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, 0, &hKey, &dwDisp);
                // 위의 키 밑에 Type이란 DWORD 타입의 값을 만들고 1로 초기화
  dwData = 0x1;
  error = RegSetValueEx( hKey, "Type", 0, REG_DWORD,&dwData,4); 
                // 위의 키 밑에 Group이란 문자열 타입의 값을 만들고 lpData의 값으로 초기화
  error = RegSetValueEx( hKey, "Group", 0, REG_SZ, lpData, strlen(lpData));

                // 키를 닫는다.
  RegCloseKey(hKey); 
 }
기존의 레지스트리 키에서 값 읽기 
 // HKEY_CURRENT_USER\Software\Netscape\Netscape Navigator\Main 밑의 Install Directory
        // 값의 문자열 값을 읽어들인다.
 DWORD dwType, cbData;
 HKEY hSubKey; 
 long lRet;
 char pszString[255];

 // 키를 오픈한다.
 if ((lRet = RegOpenKeyEx(HKEY_CURRENT_USER, 
                 "Software\\Netscape\\Netscape Navigator\\Main",
   0, KEY_READ | KEY_QUERY_VALUE , &hSubKey)) == ERROR_SUCCESS)
 {
  cbData = 255; // 문자열 값을 읽어올 데이터의 크기를 준다.
  if ((lRet = RegQueryValueEx(hSubKey, "Install Directory",
   NULL, &dwType, pszString, &cbData)) == ERROR_SUCCESS)
  {
   // 제대로 읽힌 경우
  }
  else
  {
   // 에러가 발생한 경우
  }
  RegCloseKey(hSubKey);
 }
레지스트리 키 삭제하기 - RegDeleteKey 함수를 사용한다. 

4. 윈도우 탐색기로부터의 Drag&Drop을 받으려면
윈도우 탐색기로부터 제가 만든 윈도우로의 drag&drop이 가능하게 하려면 어떻게 해야 합니까 ?

다음 순서를 따라서 프로그래밍하시면 됩니다.

프로그램의 초기화시에 DragAcceptFiles(hWnd, TRUE) 함수를 호출한다. 첫 번째 인자인 hWnd는 드롭의 타겟이 되는 윈도우의 핸들이다. 
탐색기로부터 파일이 드롭되는 순간에 WM_DROPFILES 메시지가 날라온다. 이를 처리한다. 
 case WM_DROPFILES :
 {
  POINT pt;
  // 어느 위치에 드롭되었는지 그 항목을 알아낸다.
  if (DragQueryPoint((HDROP)wParam, &pt)) 
  {
   UINT i = 0;
   // 모두 몇 개의 파일이 드롭되었는지 알아낸다.
   // 만일 폴더가 드롭되었다면 폴더의 이름만 넘어온다.
   UINT uCount = DragQueryFile((HDROP)wParam, 0xFFFFFFFF, NULL ,0);

   for(i = 0;i < uCount;i++)
   {
    // 드롭된 파일의 이름을 알아온다.
    DragQueryFile((HDROP)wParam, i, buffer ,255);
    // 드롭된 파일 이름을 출력해본다.
    MessageBox(hWnd, buffer, "File Name", MB_OK);
   }
  }
  // drag and drop 작업을 끝낸다.
  DragFinish((HDROP)wParam);
  break;
 }
Drag&drop을 더 사용할 필요가 없어지면 DragAcceptFiles를 호출한다. 
 DragAcceptFiles(hWnd, FALSE);

5. 시스템의 모든 드라이브 알아내기
현재 시스템에 붙어있는 모든 드라이브(네트웍 드라이브 포함)에 대한 정보를 알아내고 싶습니다.

GetLogicalDriveStrings로 시스템에 마운트되어있는 모든 드라이브 정보를 알아낸다. 두 번째 인자인 buffer로 드라이브 정보가 들어오는데 그 구조는 c:\,d:\과 같은 형식이며 리턴값으로 그 버퍼의 크기가 들어온다.

 char buffer[256];
 DWORD dwRet;
 LPSTR token;

 dwRet = GetLogicalDriveStrings(256, buffer);
루프를 돌면서 드라이브별 정보를 알아낸다. 이 때는 GetVolumeInformation 함수를 이용한다.

 token = buffer; // token이 지금 처리해야할 드라이브를 가리킨다.
 while (dwRet > 0)
 {
  DWORD FileSystemFlag;
  char FileSystemName[64];
    
  strcpy(DriveString, token);
  // VolumeName으로 드라이브에 대한 설명 문자열이 넘어온다.
  if (GetVolumeInformation(token, VolumeName, 255, NULL, NULL, 
                          &FileSystemFlag, FileSystemName, 63))
  {
         // 원하는 작업을 수행한다.  
  }
  dwRet -= (strlen(token)+1);
  token = token + strlen(token)+1; // 다음 드라이브로 진행한다.
 }

6. 드라이브/디렉토리/파일의 이미지 리스트 인덱스 얻기
특정 드라이브/디렉토리/파일이 시스템 이미지 리스트에서 어떤 인덱스를 갖는지 알고 싶습니다.

각 파일이나 드라이브 및 디렉토리에 대한 정보는 Shell 라이브러리에서 제공해주는 SHGetFileInfo 함수를 이용하면 됩니다. 다음의 함수는 첫 번째 인자인 lpFileName으로 주어진 파일에 대한 설명을 두 번째 인자로 받아오고 세 번째 인자로는 시스템 이미지 리스트에서의 인덱스를 얻어옵니다.

 void GetFileInfo(LPSTR lpFileName, LPSTR lpDesc, int *nIndex)
 {
     DWORD dwAttr;
     SHFILEINFO sfi;

     int hIcon = SHGetFileInfo(lpFileName, dwAttr, &sfi, sizeof(SHFILEINFO), 
    SHGFI_TYPENAME | SHGFI_SYSICONINDEX); 
     *nIndex = sfi.iIcon;
     strcpy(lpDesc, sfi.szTypeName);
 }

7. 리스트 컨트롤에 칼럼 헤더 넣기
리포트뷰 형식의 리스트 컨트롤에 컬럼 헤더를 집어 넣으려면 어떻게 해야합니까 ?


 // <문서명, 등록날짜, 상태> : 3개의 헤더를 만든다.
 LV_COLUMN col;

 col.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH;
 col.fmt = LVCFMT_LEFT;
 col.cx = 100;
 col.pszText = "문서명";
 col.cchTextMax = strlen(col.pszText);
 ListView_SetColumn(hListView, 0, &col);

 col.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH;
 col.fmt = LVCFMT_LEFT;
 col.cx = 100;
 col.pszText = "등록날짜";
 col.cchTextMax = strlen(col.pszText);
 ListView_InsertColumn(hListView, 0, &col);

 col.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH;
 col.fmt = LVCFMT_LEFT;
 col.cx = 100;
 col.pszText = "상태";
 col.cchTextMax = strlen(col.pszText);
 ListView_InsertColumn(hListView, 1, &col); 

8. 리스트뷰에 항목 삽입하기
리스트뷰에 한 항목을 추가하고 싶습니다.


 // 이미지 리스트와 부가 정보를 사용하지 않는 리스트뷰 컨트롤이다.
 int nIndex;
 LV_ITEM item;

 // - 첫번째 컬럼 -
 item.mask = LVIF_TEXT; // 이미지 리스트를 사용하려면 LVIF_IMAGE를 추가하고
                               // 부가정보를 지정해야할 일이 있다면 LVIF_PARAM을 추가한다.
 item.pszText = lpDocName;
 item.cchTextMax = strlen(lpDocName);
 item.iItem = 1;
 item.iSubItem = 0;
 nIndex = ListView_InsertItem(hListView, &item); 
 // - 두번째 컬럼 -
 item.mask = LVIF_TEXT;
 item.iItem   = nIndex;
 item.pszText = lpDate;
 item.cchTextMax = strlen(lpDate);
 item.iSubItem = 1;
 ListView_SetItem(hListView, &item); 
 // - 세번째 컬럼 -
 item.mask = LVIF_TEXT;
 item.iItem   = nIndex;
 item.pszText = "";
 item.cchTextMax = strlen(lpDocName);
 item.iSubItem = 2;
 ListView_SetItem(hListView, &item); 

9. 리스트뷰 컨트롤에서의 정렬 구현
리스트뷰 컨트롤에서 칼럼 헤더를 눌렀을 때 정렬이 되도록 하려면 어떻게 해야합니까 ?


일단 리스트뷰 컨트롤의 생성시 윈도우 스타일로 LVS_NOSORTHEADER를 주지 않는다. 
리스트뷰로부터 칼럼 헤더가 눌렸을 때 오는 이벤트를 받아들인다.

 NM_LISTVIEW *pnmtv = (NM_LISTVIEW FAR *)lParam;

 switch(pnmtv->hdr.code)
 {
  case LVN_COLUMNCLICK :
  {
   // 어느 항목(pnmtv->iSubItem)이 눌렸는지부터 검사한다. 
   // g_iSubItem은 어느 항목이 눌렸는지 기록해두는 인덱스이다.
   g_iSubItem = pnmtv->iSubItem;
   // 정렬함수를 호출한다. CompareFunc가 정렬함수이다.
   ListView_SortItems(hListView, (PFNLVCOMPARE)CompareFunc, (LPARAM)this); 
   break;
  }
리스트뷰 항목을 정렬하는데 사용되는 CompareFunc라는 함수를 만든다. 이는 보통 C 함수로 만들거나 클래스를 사용할 경우에는 클래스 내의 static 함수로 만든다. CompareFunc의 코드는 다음과 같다.

 int CALLBACK CompareFunc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
 {
  LV_FINDINFO lvfi;
  int iFirstItem, iSecondItem;

  lvfi.flags = LVFI_PARAM;
  lvfi.lParam = lParam1;
  iFirstItem = ListView_FindItem(hListWnd, -1, &lvfi);

  lvfi.flags = LVFI_PARAM;
  lvfi.lParam = lParam2;
  iSecondItem = ListView_FindItem(hListWnd, -1, &lvfi);

  char lpFirst[100];
  char lpSecond[100];
  ListView_GetItemText(hListWnd, iFirstItem, g_iSubItem, lpFisrt, 100);
  ListView_GetItemText(hListWnd, iSecondItem, g_iSubItem, lpSecond, 100);

  // g_iSubItem 컬럼의 성격에 따라 비교한다. 문자열이라면 아래와 같이 한다.
  int iRet = strcmpi(lpFirst, lpSecond); 
  return iRet;
 }

10. 버전 정보 알아내기 코드
파일의 버전을 API를 통해 알아내려면 어떻게 해야합니까 ?

Resource의 한 타입으로 VERSIONINFO라는 것이 존재합니다. 여기에 해당 파일의 버전 정보를 기록하도록 되어있습니다. 이 버전 정보를 읽어오는데 ver.dll이라는 DLL에 들어있는 API들을 사용합니다. 주의할 점은 버전 리소스는 언어별로 설정이 되기 때문에 영어로도 읽어보고 한국어 로도 읽어봐야 한다는 것입니다. 다음 예제를 참고하시기 바랍니다.

 // szDrvName이란 파일에 들어있는 버전 정보를 읽어온다. 
 #include

 DWORD      dwSize, handle;
 LPSTR      lpstrVffInfo;
 HANDLE     hMem;
 LPSTR      lpVersion;    // 이 변수로 파일의 버전 정보가 들어온다.
 char       szDrvName[80];  // 버전 정보를 알아내고자 하는 파일 이름이 여기에 들어온다.

 ....
 // 버전 정보 블록의 크기를 알아온다.
 dwSize = GetFileVersionInfoSize(szDrvName , &handle);
 if (dwSize) // 버전 정보 블록이 존재하면
 { 
  // 버전 정보 블록을 포함할 메모리 블록을 할당 받아둔다.
  hMem = GlobalAlloc(GMEM_MOVEABLE, dwSize);
  lpstrVffInfo  = GlobalLock(hMem);
  // 버전 정보 블록의 내용을 읽어온다. 
  GetFileVersionInfo(szDrvName, handle, dwSize, lpstrVffInfo);
         // 버전 정보 블록에서 버전 정보를 읽어온다.
         VerQueryValue((LPVOID)lpstrVffInfo, 
                        (LPSTR)"\\StringFileInfo\\041204B0\\FileVersion",
                        (void FAR* FAR*)&lpVersion, (UINT FAR *)&dwSize);
  // lpVersion에 들어있는 버전 정보를 사용한다.
  ....
  GlobalUnlock(hMem);
  GlobalFree(hMem);
    }
위에서 041204B0가 바로 버전 리소스에 사용된 언어가 무엇인지를 나타냅니다. 이는 영어를 나타내며 한국어의 경우에는 040904B0를 사용하면 됩니다. 이 밖에도 version.lib를 링크의 라이브러리 항목에 추가해야 합니다. 

11. 시스템 사양 알아내기
현재 시스템에 부착되어 있는 메인 메모리의 양과 CPU와 운영체제의 종류를 알고 싶습니다.

먼저 시스템에 부착되어 있는 메인 메모리의 크기는 GlobalMemoryStatus라는 API를 이용하면 됩니다. 예제 코드는 다음과 같습니다.

 //===========================================================
 // lMemTotal      : 실제 메모리의 전체 크기 (KB 단위)
 // lAvailMemTotal : 사용 가능한 실제 메모리의 크기 (KB 단위)
 // lVirtualTotal  : 가상 메모리의 전체 크기  (KB 단위)
 //===========================================================
 void GetMemoryStatus(long *lMemTotal, long *lAvailMemTotal, long *lVirtualTotal)
 {
  double var;
  MEMORYSTATUS memoryStatus;

  memset (&memoryStatus, sizeof (MEMORYSTATUS), 0);
  memoryStatus.dwLength = sizeof (MEMORYSTATUS);

  GlobalMemoryStatus (&memoryStatus);

  lMemTotal = memoryStatus.dwTotalPhys / 1024;
  lAvailMemTotal = memoryStatus.dwAvailPhys / 1024;
  lVirtualTotal = memoryStatus.dwTotalVirtual / 1024;
 }
다음으로 CPU의 종류를 알아내는 코드는 다음과 같습니다. 
 //===============================================================
 // GetProcessorInfo : 프로세서에 대한 정보를 읽어온다.
 // lpCPUSpeed      : CPU의 속도. 기록된 시스템에서만 읽어온다.
 // lpProcessorType : 프로세서의 종류
 // lpNumProcessors : 프로세서의 개수. NT의 경우에만 의미가 있다.
 //===============================================================
 void GetProcessorInfo(LPSTR lpCPUSpeed, LPSTR lpProcessorType, LPSTR lpNumProcessors)
 {
  SYSTEM_INFO sysInfo;
  LONG result;
  HKEY hKey;
  DWORD data;
  DWORD dataSize;

  lpCPUSpeed[0] = 0;
  // ---------------------------------------------
  // 프로세서의 속도를 얻어낸다.
  // ---------------------------------------------
  result = ::RegOpenKeyEx (HKEY_LOCAL_MACHINE,
   "Hardware\\Description\\System\\CentralProcessor\\0", 0, KEY_QUERY_VALUE, &hKey);
  if (result == ERROR_SUCCESS) 
  {
   result = ::RegQueryValueEx (hKey, "~MHz", NULL, NULL,(LPBYTE)&data, &dataSize);
   wsprintf(lpCPUSpeed, "%d MHz", data);
  }
  RegCloseKey (hKey);

  // ------------------------------------------
  // 하드웨어 정보를 얻어낸다.
  // ------------------------------------------
  GetSystemInfo (&sysInfo);

  // 프로세서 타입부터 검사한다.
  if (sysInfo.dwProcessorType  == PROCESSOR_INTEL_386)
   strcpy(lpProcessorType,  "Intel 386");
  else if (sysInfo.dwProcessorType  == PROCESSOR_INTEL_486)
   strcpy(lpProcessorType,  "Intel 486");
  else if (sysInfo.dwProcessorType  == PROCESSOR_INTEL_PENTIUM)
  {
   if (sysInfo.wProcessorLevel == 6) 
    strcpy(lpProcessorType, "Intel Pentium (II/Pro)");
   else
    strcpy(lpProcessorType,  "Intel Pentium");
  }
  else 
   strcpy(lpProcessorType, "알 수 없는 시스템");

  // 프로세서의 갯수를 검사한다.
  wsprintf(lpNumProcessors, "%d", sysInfo.dwNumberOfProcessors);
 }

다음으로 현재 사용 중인 운영체제의 종류를 알아내는 코드는 다음과 같습니다. 
 //===============================================================
 // GetOSVersion : OS의 버전을 얻어온다.
 // --------------------------------------------------------------
 // lpstInfo
 // lpstBuildNumber
 // lpstServicePack
 //===============================================================
 void GetOSVersion (LPSTR lpstInfo, LPSTR lpstBuildNumber, LPSTR lpstServicePack)
 {
  int stat = 0;
  TCHAR data [64];
  DWORD dataSize;
  DWORD win95Info;
  OSVERSIONINFO versionInfo;
  HKEY hKey;
  LONG result;

  lpstServicePack[0] = 0;
  versionInfo.dwOSVersionInfoSize = sizeof (OSVERSIONINFO);

  // 버전 정보를 얻어낸다.
  if (!::GetVersionEx (&versionInfo)) 
  {
   strcpy(lpstInfo, "운영체제 정보를 얻을 수 없습니다.");
   return;
  }

  // NT이면 서버인지 웍스테이션인지 검사한다. 이는 레지스트리를 보고 검사한다.
  if (versionInfo.dwPlatformId == VER_PLATFORM_WIN32_NT) 
  {
   strcpy(lpstInfo, "Windows NT");
   dataSize = sizeof (data);  
   result = ::RegOpenKeyEx (HKEY_LOCAL_MACHINE,
    "System\\CurrentControlSet\\Control\\ProductOptions", 0, KEY_QUERY_VALUE, &hKey);
   if (result != ERROR_SUCCESS) 
    return;

   result = ::RegQueryValueEx (hKey, "ProductType", NULL, NULL, (LPBYTE) data, &dataSize);
   RegCloseKey (hKey);

   if (result != ERROR_SUCCESS) 
    return;

   if (lstrcmpi (data, "WinNT") == 0) 
    strcpy(lpstInfo, "Windows NT Workstation");
   else if (lstrcmpi (data, "ServerNT") == 0) 
    strcpy(lpstInfo, "Windows NT Server");
   else 
    strcpy(lpstInfo, "Windows NT Server - Domain Controller");

   // NT 버전을 알아낸다.
   if (versionInfo.dwMajorVersion == 3 || versionInfo.dwMinorVersion == 51) 
    strcat(lpstInfo, " 3.51");
   else if (versionInfo.dwMajorVersion == 5) // 윈도우 2000의 경우
    strcat(lpstInfo, " 5.0");
   else 
    strcat(lpstInfo, " 4.0");

   // Build 번호를 알아낸다.
   wsprintf(lpstBuildNumber, "%d", versionInfo.dwBuildNumber);
  }
  else if (versionInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS) 
  {
   strcpy(lpstInfo, "Windows 95");
   if ((versionInfo.dwMajorVersion > 4) || ((versionInfo.dwMajorVersion == 4)
    && (versionInfo.dwMinorVersion > 0))) 
   {
    strcpy(lpstInfo, "Windows 98");
   }
   // 윈도우 95는 Build 번호가 하위 워드에 들어간다.
   win95Info = (DWORD)(LOBYTE(LOWORD(versionInfo.dwBuildNumber)));
   wsprintf(lpstBuildNumber, "%d", win95Info);
  }
  else 
   wsprintf(lpstInfo, "Windows 3.1");

  // 서비스 팩 정보를 얻어낸다.
  strcpy(lpstServicePack, versionInfo.szCSDVersion);
 }

12. IE의 설치 여부와 버전 확인
현재 시스템에 IE가 설치되었는지 여부와 그 버전을 알려면 어떻게 해야합니까 ?

사실 동작시켜보지 않고서는 IE가 제대로 설치되어있는지 알아내는 방법은 없지만 레지스트리를 통해 IE가 설치되었는지 여부와 버전을 확인할 수 있습니다. 그 함수는 다음과 같습니다.


 //===========================================================================
 // GetIEVersion : IE의 버전을 얻는다. 정보를 찾을 수 없으면 FALSE를 리턴한다.
 //===========================================================================
 BOOL GetIEVersion(LPSTR lpVer)
 { 
  LONG result;
  HKEY hKey;
  DWORD dwType; 
  char data[65];
  DWORD dataSize = 64;

  // --------------------
  // IE의 버전을 얻는다.
  // --------------------
  result = ::RegOpenKeyEx (HKEY_LOCAL_MACHINE, "Software\\Microsoft\\Internet Explorer", 0, KEY_QUERY_VALUE, &hKey);
  if (result == ERROR_SUCCESS) 
  {
   result = ::RegQueryValueEx (hKey, "Version", NULL, &dwType, (unsigned char *)data, &dataSize);
   strcpy(lpVer, data);
  }
  else
   return FALSE;

  RegCloseKey (hKey);
  return TRUE;
 }

13. IE의 보안 설정 보기
IE에 보면 네 단계의 보안 영역이 있습니다. 그 영역별로 설정되어있는 보안 설정값을 읽으려면 어떻게 해야합니까 ?

IE에는 다음과 같은 네 가지 보안 영역이 존재합니다.

인터넷 영역 
로컬 인터넷 영역 
신뢰할 수 있는 사이트 영역 
제한된 사이트 영역 
IE는 보안 영역 설정과 관련하여 Internet Security Manager와 Internet Zone Manager라는 인터페이스가 존재합니다. 이를 이용해 보안 영역의 보안을 설정하고 특정 IP나 도메인 이름을 등록할 수 있습니다. 자세한 사항은 레퍼런스를 찾아보기 바랍니다. 
 
 #include "objbase.h"
 #include "urlmon.h"

 char szTemp1[256];
 char szTemp2[256];
 HRESULT hr;
 IInternetSecurityManager *pSecurityMgr;
 IInternetZoneManager *pZoneMgr;
 DWORD dwEnum, dwZoneCount;
 // --- 변수 선언부
 DWORD dwZone;
 ZONEATTRIBUTES zoneAttr;
 int nLevel = 2;

 // COM 라이브러리를 초기화한다.
 CoInitialize(NULL);
 dwEnum = 0;
 pSecurityMgr = NULL;
 pZoneMgr = NULL;

 // Internet Security 인터페이스 초기화
 hr = CoCreateInstance(CLSID_InternetSecurityManager, NULL, CLSCTX_ALL, //INPROC_SERVER,
   IID_IInternetSecurityManager, (void**)&pSecurityMgr);
 if (hr != S_OK)
 {
  return;
 }
 hr = CoCreateInstance(CLSID_InternetZoneManager, NULL, CLSCTX_ALL, //INPROC_SERVER,
   IID_IInternetZoneManager, (void**)&pZoneMgr); 
 if (hr != S_OK)
 {
  return;
 }
 dwEnum = 0;

 // 보안 영역 열거자(Zone Enumerator)를 초기화한다.
 pZoneMgr->CreateZoneEnumerator(&dwEnum, &dwZoneCount, 0);
 for(DWORD i = 1;i < dwZoneCount;i++)
 {
  pZoneMgr->GetZoneAt(dwEnum, i, &dwZone);
  pZoneMgr->GetZoneAttributes(dwZone, &zoneAttr);

  // zoneAttr.szDisplayName에 보안 영역의 이름이 들어오는데 유니코드이다. 이를 변환한다.
  WideCharToMultiByte(CP_ACP, 0, zoneAttr.szDisplayName, -1, szTemp1, 255, NULL, NULL); 
  // zoneAttr.dwTemplateCurrentLevel에는 보안 영역의 보안값 설정이 들어온다.
  wsprintf(szTemp2, "%x", zoneAttr.dwTemplateCurrentLevel);
 }

 // 보안 영역 열거자(Zone Enumerator)를 제거한다.
 if (dwEnum != 0)
  pZoneMgr->DestroyZoneEnumerator(dwEnum);

 pSecurityMgr->Release();
 pZoneMgr->Release();
 // COM 라이브러리를 메모리에서 내린다.
 CoUninitialize();
}

14. ActiveX 컨트롤의 등록 방법
Regsvr32 같은 유틸리티를 이용하지 않고 프로그램 내에서 컨트롤을 레지스트리에 등록하려면 어떻게 해야합니까 ?

모든 AcitveX 컨트롤은 자신을 레지스트리에 등록하기위한 목적으로 DllRegisterServer라는 함수를 갖고 있습니다. ActiveX 컨트롤을 메모리로 로드한 다음에 이 함수를 불러주면 원하는 일을 수행할 수 있습니다. 반대로 ActiveX 컨트롤을 레지스트리에서 제거하기 위한 용도로 DllUnRegisterServer라는 함 수도 존재합니다.


 // ==============================================================
 // RegisterOCX     지정된 ActiveX 컨트롤을 레지스트리에 등록한다.
 // --------------------------------------------------------------
 // LPSTR pszString 등록하고자 하는 ActiveX 컨트롤의 절대 경로명
 // ==============================================================
 BOOL WINAPI RegisterOCX(LPSTR pszString)
 {
  int iReturn = 0;
  HRESULT (STDAPICALLTYPE * lpDllEntryPoint)();
  HINSTANCE hLib;

  // OLE 라이브러리를 초기화한다.    
  if (FAILED(OleInitialize(NULL)))
  {
   MessageBox(GetFocus(), "OLE 초기화 실패", "에러", MB_OK);
   return FALSE;
  }

  // 지정된 activeX 컨트롤을 메모리로 로드한다.
  hLib = LoadLibrary(pszString);
  if (hLib <= NULL)
  {
   MessageBox(GetFocus(), "파일을 로드하는데 실패했습니다.", "에러", MB_OK);
   OleUninitialize();
   return FALSE;
  }

  // "DllRegisterServer" 함수의 위치를 찾는다.
  lpDllEntryPoint = (long (__stdcall *)(void))GetProcAddress(hLib, "DllRegisterServer");
 
  // 이 함수를 호출합니다.
  if (lpDllEntryPoint)
  {
   if (FAILED((*lpDllEntryPoint)()))
   {
    DWORD dwRet;
    char szTemp[128];

    dwRet = GetLastError();
    wsprintf(szTemp, "에러 번호 : %lx", dwRet);
    MessageBox(GetFocus(), szTemp, "DllRegisterServer 에러", MB_OK);
    FreeLibrary(hLib);
    OleUninitialize();
    return FALSE;
   }
  }
  else
  {
   MessageBox(GetFocus(), "DllRegisterServer를 찾을 수 없습니다.", "에러", MB_OK);
   FreeLibrary(hLib);
   OleUninitialize();
   return FALSE;
  }

  FreeLibrary(hLib);
  OleUninitialize();
  return TRUE;
 }

15. 데이터 파일의 실행
탐색기에서 실행 파일이 아닌 데이터 파일을 더블클릭하면 그 데이터 파일과 연결된 실행 파일이 실행되면서 그 파일을 물고 올라갑니다. 이를 구현하는 방법을 알려주세요.

ShellExecuteEx라는 API를 사용하면 됩니다. 다음 함수는 인자로 데이터 파일 혹은 실행 파일의 경로를 받아서 실행해줍니다. 데이터 파일의 경우에는 연결된 실행 파일을 찾아서 그걸 실행해줍니다.

 BOOL Execute(LPSTR lpPath)
 {
  char FilePath[255];
  SHELLEXECUTEINFO  ExecInfo;
 
  // lpPath를 나누어 본다.
  char drive[_MAX_DRIVE];
  char dir[_MAX_DIR];
  char fname[_MAX_FNAME];
  char ext[_MAX_EXT];

  _splitpath(lpPath, drive, dir, fname, ext);
  // 디렉토리 경로를 얻는다.
  strcpy(FilePath, drive);
  strcat(FilePath, dir);

  // 파일 이름을 얻는다.
  strcat(fname, ".");
  strcat(fname, ext);

  SHELLEXECUTEINFO  ExecInfo;
  ExecInfo.cbSize = sizeof(SHELLEXECUTEINFO); 
  ExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS; 
  ExecInfo.hwnd = hWnd; 
  ExecInfo.lpVerb = "open"; 
  ExecInfo.lpFile = fname; 
  ExecInfo.lpParameters = NULL; 
  ExecInfo.lpDirectory = FilePath; 
  ExecInfo.nShow = SW_SHOW; 
  ExecInfo.hInstApp = g_hInstance; // g_hInstance는 프로그램의 인스턴스 핸들

  return ShellExecuteEx(&ExecInfo);
 }

16. 파일의 존재 여부 테스트 
어떤 파일이 실제로 존재하는 것인지 간단히 테스트해보는 방법은 무엇인가요 ?

CreateFile API를 사용하면 됩니다. 파일을 열때 플래그 중의 하나로 OPEN_EXISTING이라는 것이 있는데 이를 사용하면 됩니다. 다음 코드를 예로 보시기 바랍니다.


 hFile = CreateFile("C:\\TEMP\\test.txt", GENERIC_READ, 0, NULL,  
   OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
 if (hFile != INVALID_HANDLE_VALUE)
 {
  // 파일이 존재하는 경우
  CloseHandle(hFile);
 }

17. API를 이용한 파일 I/O
API를 이용한 파일 입출력에 대한 예제 코드가 없습니까 ?

프로그래밍을 하다보면 간혹 파일 I/O를 API를 이용해 수행해야 할 경우가 있습니다. 이 때 다음의 예제 코드를 복사해다가 사용하면 편리할 것입니다. MFC의 CFile을 이용한 파일 I/O에 대해 알고 싶으시면 요기를 클릭하세요.

파일 쓰기의 경우 
 HANDLE fp;
 DWORD NumberOfBytesWritten;
 char lpBuffer[1024];

 // FileName이 지정한 파일의 이름이 있으면 그걸 열고 없으면 그 이름으로 하나 생성한다.
 if ((fp=CreateFile((LPCTSTR)FileName, GENERIC_WRITE | GENERIC_READ, 0, NULL, 
      OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL)) == INVALID_HANDLE_VALUE)
 {
  // 파일 열기 에러 발생
 }  
 // 필요한 만큼 WriteFile의 호출을 반복한다. 파일 포인터의 이동시에는 SetFilePointer API를 이용한다.
 WriteFile(fp, lpBuffer, 1024, &NumberOfBytesWritten, NULL);
 if (NumberOfBytesWritten != 1024)
 {
  // 파일 쓰기 에러 발생 
  CloseHandle(fp);  
 }
 // 작업이 다 끝났으면 파일을 닫는다.
 CloseHandle(fp);  
파일 읽기의 경우 
 HANDLE fp;
 DWORD NumberOfBytesRead;
 char lpBuffer[1024];

 if ((fp=CreateFile((LPCTSTR)FileName, GENERIC_READ, 
  FILE_SHARE_READ, NULL, OPEN_EXISTING, 
  FILE_ATTRIBUTE_NORMAL, NULL)) == INVALID_HANDLE_VALUE)
 {
  // 파일 열기 에러 발생
 }  
 // 필요한 만큼 ReadFile의 호출을 반복한다.
 ReadFile(fp, lpBuffer, 1024, &NumberOfBytesRead, NULL);
 if (NumberOfBytesRead != 1024)
 {
  // 파일 읽기 에러 발생 
  CloseHandle(fp);  
 }
 // 작업이 다 끝났으면 파일을 닫는다.
 CloseHandle(fp);  
파일 포인터의 이동시에는 SetFilePointer라는 API를 사용하고 파일의 크기를 알고 싶을 때는 GetFileSize라는 API를 사용한다.


18. GetParent API의 리턴값
다이얼로그 박스에서 GetParent API를 호출했을 때 리턴되는 값이 이상합니다.

GetParent API의 리턴값은 보통의 윈도우에서는 윈도우 생성시 지정한 부모/자식 윈도우 간의 관계에 따라 달라집니다. 하지만 다이얼로그 박스의 경우에는 다릅니다. 다이얼로그 박스는 GetParent를 호출하면 무조건 그 것이 속한 응용프로그램의 메인 윈도우 핸들이 리턴됩니다.


19. 특정 프로그램을 실행하고 종료를 기다리기
특정 프로그램을 실행한 다음에 그 프로그램이 종료될 때 다른 일을 하고 싶습니다. 어떻게 해야합니까 ?

다음은 CreateProcess를 이용해서 특정 프로그램의 실행이 끝나기를 기다리는 코드입니다.

 // buffer에 실행하고자하는 파일이름이 들어온다.
 void DoCreate(HWND hWnd, LPSTR buffer)
 {
  STARTUPINFO            sui;
  PROCESS_INFORMATION    pi;
  DWORD                  ret;

  memset(&sui, 0x00, sizeof(STARTUPINFO));
  sui.cb = sizeof(STARTUPINFO);
    
  ret = CreateProcess(buffer, NULL, NULL, NULL, FALSE, 
   0, NULL, NULL,&sui, &pi);
  if (ret == TRUE) // 제대로 실행되었으면
  {
   hProcess = pi.hProcess;
   // 실행이 끝나기를 대기한다.
   WaitForSingleObject(hProcess, 0xffffffff);
   CloseHandle(hProcess);
  }
 }


20. 파일 열기 다이얼로그 띄우기
API를 이용해 파일 오픈 다이얼로그를 띄우고 싶습니다.

API를 이용해 파일 오픈 다이얼로그를 띄우는 방법은 다음과 같습니다.

 #define MAXCHARS   255
 OPENFILENAME       ofn;
 char               buffer[MAXCHARS];

 memset(&ofn, 0x00, sizeof(OPENFILENAME));
 ofn.lStructSize = sizeof(OPENFILENAME);
 ofn.hwndOwner = hWnd;
 ofn.lpstrFilter = "모든 파일\000*.*\000\000";
 ofn.lpstrInitialDir = m_szTemp;
 ofn.nFilterIndex = 1;
 ofn.lpstrFile = buffer;
 ofn.nMaxFile = MAXCHARS;
 ofn.lpstrTitle = "파일 선택하기";

 if (GetOpenFileName(&ofn))  // 사용자가 파일을 제대로 선택한 경우
 {
  // ....
 }

21. 긴 파일 이름과 짧은 파일 이름간의 변환 방법
주어진 긴 파일 이름에 해당하는 짧은 파일 이름을 알려면 어떻게 해야하나요 ?

긴 파일 이름에서 짧은 파일 이름으로의 변환 : GetShortPathName API 사용 
짧은 파일 이름에서 긴 파일 이름으로의 변환 : GetFullPathName API 사용 

22. 시스템 이미지 리스트 얻어내기
탐색기 등에서 사용되는 시스템 이미지 리스트를 사용할 수 있는 방법이 있는지 알고 싶습니다.

물론 있습니다. SHGetFileInfo라는 API를 이용하면 됩니다. 이를 이용하면 16X16이나 32X32 크기의 이미지 리스트를 얻어낼 수 있습니다.

 SHFILEINFO sfi;
 HIMAGELIST sysSmallList, sysLargeList;

 sysSmallList = (HIMAGELIST)SHGetFileInfo(TEXT("C:\\"), 0, &sfi,   sizeof(SHFILEINFO), SHGFI_SYSICONINDEX | SHGFI_SMALLICON);
 sysLargeList = (HIMAGELIST)SHGetFileInfo(TEXT("C:\\"), 0, &sfi,   sizeof(SHFILEINFO), SHGFI_SYSICONINDEX | SHGFI_ICON);

23. 리스트뷰 컨트롤에서 스타일 동적 변경하기
리스트뷰 컨트롤에서 리스트(LVS_ICON) 스타일을 사용 중인데 이를 실행 중에 리포트(LVS_REPORT) 스타일로 변경하고 싶습니다. 어떻게 해야할까요 ?

윈도우의 스타일은 윈도우 엑스트라 바이트라는 영역에 저장됩니다. 이 곳의 데이터를 읽고 쓰는데 GetWindowLong, SetWindowLong 같은 API를 사용하는데 다음 코드를 예로 보시기 바랍니다.

리스트 스타일에서 리포트 스타일로 
 LONG lStyle = GetWindowLong(hListWnd, GWL_STYLE);
 SetWindowLong(hListWnd, GWL_STYLE, (lStyle & ~LVS_LIST) | LVS_REPORT);
리포트 스타일에서 리스트 스타일로 
 LONG lStyle = GetWindowLong(hListWnd, GWL_STYLE);
 SetWindowLong(hListWnd, GWL_STYLE, (lStyle & ~LVS_REPORT) | LVS_LIST);

24. 윈도우와 다이얼로그에서 클래스 포인터의 사용
제가 만든 C++ 클래스내에서 윈도우를 생성합니다. 생성한 윈도우의 윈도우 프로시저는 그 클래스의 정적 멤버 함수로 선언되어 있습니다. 정적 함수의 경우에는 데이터 멤버를 접근하지 못하기 때문에 접근하게 하려고 윈도우를 생성한 그 클래스의 객체에 대한 포인터를 전역 변수로 유지하여 사용하고 있습니다. 별로 깨끗한 방법도 아닌 것 같고 또 동시에 여러 개의 객체가 동작할 경우에는 에러가 날 수밖에 없는데 해결 방법이 없을까요 ?

이는 다이얼로그의 경우에도 마찬가지입니다. 윈도우 생성시 사용하는 CreateWindow나 CreateWindowEx 같은 함수를 보면 마지막 인자로 윈도우 생성 데이터라는 것을 지정하게 되어있습니다. 다이얼로그 생성시에는 DialogBox라는 API말고 DialogBoxParam이라는 API가 있어서 마지막 인자로 초기화 데이터를 넘겨줄 수 있도록 되어있는데 이것과 윈도우 엑스트라 바이트를 같이 사용하면 정적함수로 선언된 윈도우 프로시저나 다이얼로그박스 프로시저내에서 객체에 대한 포인터를 사용할 수 있습니다. 예를 들어 살펴보겠습니다.

윈도우의 경우 
다음과 같이 CExplorerBar라는 클래스내에서 윈도우를 하나 생성합니다. CreateWindowEx 함수나 CreateWindow 함수의 마지막 인자로 this 포인터를 지정합니다.

 BOOL CExplorerBar::RegisterAndCreateWindow(void)
 {
  ....
  CreateWindowEx( 0,  EB_CLASS_NAME, NULL,
                     WS_CHILD | WS_CLIPSIBLINGS | WS_BORDER,
                     rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top,
                     m_hwndParent, NULL, g_hInst, (LPVOID)this);
      
   }
위에서 생성한 윈도우의 윈도우 프로시저는 WndProc이고 CExplorerBar 클래스의 정적 멤버 함수로 존재한다고 가정하겠습니다.

 LRESULT CALLBACK CExplorerBar::WndProc(HWND hWnd, UINT uMessage, WPARAM wParam, LPARAM lParam)
 {
  CExplorerBar  *pThis = (CExplorerBar*)GetWindowLong(hWnd, GWL_USERDATA);

  switch (uMessage)
  {
   case WM_NCCREATE:
   {
    LPCREATESTRUCT lpcs = (LPCREATESTRUCT)lParam;
    pThis = (CExplorerBar*)(lpcs->lpCreateParams);
    SetWindowLong(hWnd, GWL_USERDATA, (LONG)pThis);
   }
   break;
   case WM_CREATE :
    return pThis->OnCreate();
WM_NCCREATE 메시지에서 lParam 인자로 넘어오는 CExplorerBar 객체에 대한 포인터를 받아서 윈도우 엑스트라 바이트로 저장하고 있습니다. 윈도우 엑스트라 바이트는 윈도우 마다 할당되는 고유의 영역으로 사용자 정의 영역으로 GWL_USERDATA가 정의되어 있습니다. 여기에다 CExplorerBar 객체에 대한 포인터를 저장해두고 윈도우 프로시저에 진입할 때마다 이 값을 pThis라는 변수에 대입해 놓고 사용합니다. 참고로 WM_NCCREATE는 WM_CREATE 메시지보다 먼저 발생하는 메시지입니다.

다이얼로그의 경우 
예를 들어 CKTree라는 클래스내의 한 멤버 함수에서 다이얼로그 박스를 띄운다고 가정하겠습니다.

 BOOL CKTree::SelectFolder(short sTypes, long dwFolderID, short  bCreationEnabled)
 {
  // ......
  if (DialogBoxParam(hInst, MAKEINTRESOURCE(IDD_FOLDER_SELECT), hWnd, (DLGPROC)SelectFolderDlg, (LPARAM)this))
DialogBoxParam 함수는 DialogBox 함수보다 인자가 하나 더 있는데 그것이 바로 다이얼로그 프로시저의 WM_INITDIALOG 메시지의 lParam 인자로 넘어갑니다. 여기에 CKTree 객체에 대한 포인터를 넘깁니다. 그리고나서 다이얼로그 박스 프로시저의 WM_INITDIALOG 메시지에서 이를 받아서

 LRESULT CALLBACK SelectFolderDlg(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
 {
  switch (message) 
  { 
         case WM_INITDIALOG:
   {
    // lParam으로 넘어온 값을 KPointer라는 윈도우 프로퍼티에 저장한다.
    SetProp(hDlg, "KPointer", (HANDLE)lParam);
    .......
   case  WM_NOTIFY :
   {
    // CKTree 객체에 대한 포인터가 필요하면 KPointer 윈도우 프로퍼티 값을 읽어들인다.
    CKTree *pKTree = (CKMartTree *)GetProp(hDlg, "KPointer");
    if (pKTree->m_bWait)
    {
     .....
여기서는 앞서 윈도우와는 달리 윈도우 프로퍼티라는 것을 이용해서 넘어온 포인터 정보를 저장하고 필요할 때 읽어옵니다. 여기서 앞서의 윈도우 엑스트라 바이트를 사용해도 무방합니다.


25. 특정 프린터로 출력하기
제 PC에는 두 대의 프린터가 붙어있습니다. 다이얼로그를 띄우지 않고 상황에 따라 다른 프린터로 출력하고 싶은데 동적으로 HDC를 생성하는 방법을 모르겠습니다.

CreateDC를 이용하면 됩니다. 시스템 디렉토리의 WIN.INI를 보면 [Devices]라는 섹션이 있는데 이 아래로 이 시스템에 설치되어 있는 모든 프린터 드라이버의 목록이 나옵니다. 예를 들어 다음과 같이 나옵니다.

 [Devices]
 HUNFAX=HUNFAX,FaxModem
 삼성 SLB-6216H PCL5=SSMPCL5,\\영업팀\볼륨프린터
 ......
프린터별로 DeviceName=DriverName,OutputName와 같은 구조로 구성되어 있습니다. 이 값들을 CreateDC의 인자로 사용하면 됩니다. 예를 들어 위에서 두 번째 프린터의 DC를 생성하려면 다음과 같이 CreateDC를 호출합니다.

 // hDC = CreateDC(DriverName, DeviceName, OutputName, NULL);
 hDC = CreateDC ("SSMPCL5", "삼성 SLB-6216H PCL5", "\\\\영업팀\\볼륨프린터", NULL) ;
CreateDC의 마지막 인자로는 프린터의 설정값을 변경할 수 있습니다. 예를 들어 가로로 찍는다든지 2장을 찍는다든지 하는 설정을 변경하는데 사용됩니다. NULL을 주면 디폴트 값을 사용합니다. 다른 설정을 사용하고 싶다면 다음과 같이 합니다. CreateDC 호출 앞에서 DocumentProperties라는 함수를 호출하여 프린터의 기본 설정을 읽어온 다음에 이를 변경합니다. 다음 예는 출력 방향을 가로로 변경하는 예제입니다.

 // 프린터의 디폴트 설정을 읽어온다.
 LPDEVMODE lpoutDevMode;
 HANDLE hPrinter;
 HDC hPrnDC;

 // 프린터의 핸들을 얻는다.
 if (OpenPrinter( lpDeviceName, &hPrinter,  NULL))
 {
  // OpenPrinter로 얻은 프린터의 초기 설정을 DocumentProperties API로 얻어온다.
  // 먼저 마지막 인자를 0으로 해서 DocumentProperties를 호출하여 필요한 버퍼의 크기를 알아옵니다.
  long lSize = DocumentProperties(GetFocus(), hPrinter, lpPrinterName, NULL, NULL, 0);
  lpoutDevMode = (LPDEVMODE)malloc(lSize);
  long lRet = DocumentProperties(GetFocus(), hPrinter, lpPrinterName, lpoutDevMode, NULL, DM_OUT_BUFFER);
  if (lRet == IDOK)
  {
   // 프린터의 인쇄 방향 설정을 변경한다.
   // 여기서 원하는 변환을 수행한다.
   lpoutDevMode->dmOrientation = DMORIENT_LANDSCAPE;
  }
  hPrnDC = CreateDC (lpDriverName, lpDeviceName, lpOutputName, lpoutDevMode) ;
  free(lpoutDevMode);
  return hPrnDC;
 }

26. 메뉴 관련 함수
메뉴 항목을 하나 추가하려고 합니다. InsertMenuItem API를 사용하는데 윈도우 3.1에서와 사용법이 다른 것 같습니다.

사용법이 달라졌습니다.

 MENUITEMINFO mii;

 memset(&mii, 0x00, sizeof(MENUITEMINFO));
 mii.cbSize = sizeof(MENUITEMINFO);
 mii.fMask = MIIM_TYPE;
 mii.fType = MFT_SEPARATOR;
 InsertMenuItem(hSubMenu, GetMenuItemCount(hSubMenu), TRUE,  &mii);
       
 mii.cbSize = sizeof(MENUITEMINFO);
 mii.fMask = MIIM_TYPE | MIIM_STATE | MIIM_ID;
 mii.fType = MFT_STRING;
 mii.fState = MFS_DEFAULT | MFS_UNHILITE;
 mii.wID = ID_WORKPLACE_REMOVE;
 mii.dwTypeData = "바구니에서 제거";
 InsertMenuItem(hSubMenu, GetMenuItemCount(hSubMenu), TRUE,  &mii);

27. 코드 실행 중에 다른 윈도우 메시지 처리하기 
하나의 함수 내에서 시간이 오래 걸리는 작업을 하고 있습니다. 이 때 작업 취소를 위한 별도의 다이얼로그를 하나 띄워 두었는데 이 쪽의 버튼이 눌리지 않습니다. 어떻게 해야할까요 ?

시간이 오래 걸리는 작업을 별도의 스레드로 만들어 처리하던지 아니면 시간이 오래 걸리는 작업을 수행하는 함수 안에서 다음 코드를 가끔 호출해주면 됩니다. 만일 루프를 돌고 있다면 루프내에서 한번씩 호출해주면 됩니다.

 MSG       msg;

 while (PeekMessage(&msg, NULL, NULL, NULL, TRUE))
 {
  TranslateMessage(&msg);
  DispatchMessage(&msg);
 }
VB에서라면 DoEvents라는 메소드를 호출해주면 됩니다.


28. 메인 윈도우에서 캡션을 제거하고 싶습니다. 어떻게 해야 합니까 ?
윈도우를 생성할 때 WS_CAPTION이란 스타일을 지정하지 않아도 항상 윈도우의 캡션이 보입니다. 이를 제거하려면 어떻게 해야 하나요 ?

캡션을 제거하려는 윈도우의 WM_NCCREATE 메시지를 처리해야 합니다. 이 메시지는 WM_CREATE 메시지보다 앞서 발생하는 메시지입니다. NC는 Non-Client를 나타냅니다. GetWindowLong과 SetWindowLong API를 이용해서 WS_CAPTION 스타일을 제거합니다. 이 두 API는 앞서 리스트뷰 컨트롤에서 스타일 동적 변경하기에서 이미 사용해본 바 있습니다.

  case WM_NCCREATE :
  {
   long lStyle;
    
   lStyle = GetWindowLong(hWnd, GWL_STYLE);
   lStyle = (lStyle & (~WS_CAPTION)); 
   SetWindowLong (hWnd, GWL_STYLE, lStyle);
   return TRUE;
  }

29. 사각형 형태 이외의 모양을 갖는 윈도우를 띄우고 싶습니다.
윈도우는 기본 모양이 사각형인데 다른 형태의 모양을 갖는 윈도우를 띄우려면 어떻게 해야합니까 ?

원하는 모양을 Region이란 것으로 만들어야 합니다. 만든 다음에 이것을 원하는 시점에 SetWindowRgn라는 API를 이용해 윈도우에 설정해주면 됩니다. 예를 들어 타원 모양의 윈도우를 띄우고 싶다면 다음과 같이 해주면 됩니다.

 HRGN g_hRgn;

 g_hRgn = CreateEllipticRgn(0, 0, 700, 600);
 SetWindowRgn(hWnd, g_hRgn, FALSE);  
이렇게 했을 경우 윈도우의 위치 이동이 문제가 됩니다. 보통 캡션을 잡고 이동시켜야 하는데 캡션이 없으니까 문제가 됩니다. 이에 관한 것은 31. 윈도우의 이동 처리하기를 참고하기 바랍니다.


30. 시스템에 설치되어 있는 모든 프린터 드라이버 알아내기
현재 시스템에 설치되어 있는 모든 프린터 드라이버의 종류를 알아내고 싶습니다.

EnumPrinters라는 API를 사용하면 됩니다. 다음 코드는 현재 시스템에 설치되어 있는 모든 프린터 드라이버(로컬과 네트웍 프린터 포함)의 이름을 메시지박스로 보여주는 예제입니다.

 BOOL bSuccess;
 DWORD cbRequired, cbBuffer, nEntries;
 PRINTER_INFO_1 *lpBuffer = NULL;

 // 버퍼의 크기를 알아낸다. cbRequired로 들어온다.
 EnumPrinters(PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL, NULL, 1, (unsigned char *)lpBuffer, 0, &cbRequired, &nEntries);
 cbBuffer = cbRequired;
 // 버퍼를 다시 버퍼를 잡는다.
 lpBuffer = (PRINTER_INFO_1 *)malloc(cbBuffer);
 // 프린터의 종류를 알아낸다.
 bSuccess = EnumPrinters(PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL, NULL, 1, (unsigned char *)lpBuffer, cbRequired, &cbRequired, &nEntries);
 if (bSuccess == FALSE)
 {
  free(lpBuffer);
  // 다른 이유로 에러가 난 경우
  return;
 }
 // 알아낸 프린터를 하나씩 enumerate한다.
 for (int i = 0;i < nEntries; i++)
 {
  ::MessageBox(NULL, lpBuffer[i].pName, "프린터 이름", MB_OK);
   }
 free(lpBuffer);

31. 윈도우의 이동 처리하기
보통 윈도우는 캡션 영역을 잡고 위치 이동을 수행하게 되는데 윈도우의 특정 영역을 잡고 이동할 수 있게 하려면 어떻게 해야합니까 ?

WM_NCHITTEST라는 메시지를 잘(?) 처리하면 어느 영역이든 윈도우 이동을 처리할 수 있습니다. 마우스로 윈도우 위를 이동하면 WM_NCHITTEST, WM_SETCURSOR, WM_MOUSEMOVE 같은 메시지들이 발생합니다. WM_NCHITTEST는 현재 마우스가 윈도우의 어느 영역위에 있는지 알아내기 위해 사용됩니다. 이 메시지의 처리부에서 HTCAPTION이란 값을 리턴해주면 윈도우 운영체제는 지금 마우스 포인터가 윈도우의 캡션 부분에 와있다고 생각해서 여기서 드래그 작업이 시작될 경우에 윈도우의 위치를 이동시켜 버립니다. 다음 코드는 WM_NCHITTEST 메시지를 처리하여 현재 마우스 좌표가 정해진 영역에 있으면 HTCAPTION을 돌려주는 예제입니다.

 case WM_NCHITTEST:
 {
  // 현재 마우스 위치를 바탕으로 pt 변수를 채운다.
  POINT pt(LOWORD(lParam), HIWORD(lParam));

  // 마우스 좌표를 윈도우의 좌측 상단 기준의 좌표로 변경한다.
  ScreenToClient(hWnd, &pt);
  // 지정된 사각형 안에 포함되는 점인지 검사한다. 
  // g_TitleRect는 RECT 타입의 변수로 지정된 사각형의 좌표가 들어있다.
  if (PtInRect(&g_TitleRect, pt))
   return HTCAPTION;   
  break;
 }


32. 바탕 화면 위의 모든 윈도우를 최소화하거나 모든 최소화 실행 취소
바탕 화면 위의 모든 윈도우를 최소화하거나 모두 최소화 실행 취소를 프로그램으로 구현하는 방법을 알고 싶습니다.

태스크바의 빈 공간을 오른쪽 마우스 버튼으로 클릭해보면 팝업 메뉴가 뜨는데 거기에 보면 "모든 창을 최소화(M)"와 "모두 최소화 실행 취소(U)" 명령이 존재하는데 그것을 대신 선택해주는 형식으로 프로그램을 작성해주면 됩니다.

다음은 "모든 창을 최소화"해주는 루틴입니다. keybd_event API를 이용해서 사용자가 키입력한 것처럼 흉내내줍니다.

 void IconizeAllWindow()
 { 
  keybd_event(0x5b, 0, 0, 0);
  keybd_event(77, 0, 0, 0);    // 'M' key
  keybd_event(0x5b, 0, 2, 0);
 }
다음은 "모두 최소화 실행 취소" 루틴입니다.

 void RestoreWindowState()
 {
  keybd_event(0x5b, 0, 0, 0);
  keybd_event(84, 0, 0, 0);    // 'U' key
  keybd_event(0x5b, 0, 2, 0);
 }

33. VxD 드라이버 호출하기
Vxd 드라이버를 동적으로 로드해서 호출하고 싶습니다. 어떻게 해야합니까 ?

먼저 해당하는 VxD 드라이버의 이름과 위치와 호출하려는 작업의 작업 코드명을 알아야 합니다. 드라이버의 로드는 CreateFile API를 이용합니다. VxD 드라이버의 호출은 DeviceIoControl API를 이용합니다. 자세한 설명은 DeviceIoControl API의 레퍼런스를 참고하기 바랍니다.

 DWORD byteReturned;

 // 먼저 VxD 드라이버를 오픈한다.
 hDevice = CreateFile("\\\\.\\NMOUSE.VXD", 0, 0, 0, CREATE_NEW, FILE_FLAG_DELETE_ON_CLOSE, 0);
 if (DeviceIoControl(hDevice, W32_SETHWND, &hWnd, 4, NULL, 0, &byteReturned, NULL))
 {
  // Success !!!
 }
Vxd 드라이버는 kernel 모드(이를 윈도우에서는 Ring 0라고 부릅니다)에서 동작하기 때문에 모든 하드웨어와 메모리를 바로 접근할 수 있습니다. VxD 드라이버를 작성하려면 DDK를 이용하거나 Numega의 DriverStudio나 KRFTech사의 WinDriver를 이용해야 합니다.


34. Thread 실행시 에러
CreateThread를 이용해 스레드를 만들어 생성하고 있습니다. 루틴에 이상은 없는 것 같은데 스레드가 많이 생성되어 시간이 좀 지나면 에러가 발생합니다. 이유가 무엇일까요 ?

정말로 스레드 코드에 별 이상이 없다면 CreateThread API 대신에 beginthread나 beginthreadex를 사용해보기 바랍니다. 자세한 사항은 마이크로소프트의 Knowledge base를 참고하시기 바랍니다.


35. 윈도우 운영체제 종료하기
프로그램에서 특정 상황이 되면 윈도우 운영체제를 종료하고 싶습니다.

ExitWindowsEx API를 사용하면 됩니다. 이 함수의 원형은 다음과 같습니다.

 BOOL ExitWindowsEx(UINT uFlags, DWORD dwReserved);
uFlags로 종료방법을 지정할 수 있습니다. 다음과 같은 값이 가능합니다.

EWX_LOGOFF 현재 사용자를 로그오프한다. 
EWX_POWEROFF 시스템을 종료하고 파워오프한다. 파워오프는 이를 지원하는 하드웨어에서만 가능하다. 
EWX_REBOOT 시스템을 종료하고 시스템을 재시동한다. 
EWX_SHUTDOWN 시스템을 종료한다. 
EWX_FORCE WM_QUERYSESSION이나 WM_ENDQUERYSESSION을 보내지 않고 실행중인 모든 프로세스를 종료한다. 위의 네 가지 플래그들과 함께 사용할 수 있다.


36. 디폴트 웹 브라우저 알아내기
디폴트로 지정된 웹 브라우저를 실행하는 방법을 알고 싶습니다.

디폴트로 지정된 웹 브라우저는 레지스트리에 자신을 등록합니다. .htm (혹은 .html) 파일의 편집기로 링크도 되지만 http, ftp, gopher 등의 프로토콜 연결 프로그램으로 등록됩니다. 제 생각에 가장 좋은 것은 http 프로토콜의 연결 프로그램을 찾아보는 것으로 생각됩니다. 다음 레지스트리 항목에 보면 연결된 웹 브라우저의 절대 경로를 알 수 있습니다.

    HKEY_CLASSES_ROOT\http\shell\open\command
이 항목의 값을 읽는 방법은 3. 레지스트리 읽기/쓰기를 참고하고 프로그램의 실행에 관한 부분은 19. 특정 프로그램을 실행하고 종료를 기다리기를 참고하거나 ShellExecute API 혹은 WinExec API를 사용하면 됩니다. 이 기능을 수행하는 함수는 다음과 같습니다.

void LaunchDefaultWebBrowser(HWND hWnd)
{
    // HKEY_CLASSES_ROOT\http\shell\open\command
 DWORD dwType, cbData;
 HKEY hSubKey; 
 long lRet;
 LPSTR pszString, pszSrcPath;

 // 키를 오픈한다.
 if ((lRet = RegOpenKeyEx(HKEY_CLASSES_ROOT, "http\\shell\\open\\command",
   0, KEY_READ | KEY_QUERY_VALUE , &hSubKey)) == ERROR_SUCCESS)
 {
  cbData = 255; // 문자열 값을 읽어올 데이터의 크기를 준다.
        pszString = (LPSTR)malloc(255);
        pszSrcPath = pszString;
  if ((lRet = RegQueryValueEx(hSubKey, "",
   NULL, &dwType, (unsigned char *)pszString, &cbData)) == ERROR_SUCCESS)
  {
   // pszString에 디폴트 웹 브라우저의 경로가 들어온다.
            // pszString에서 "를 제거한다.
            RemoveChar(pszString, '"');
            WinExec(pszString, SW_SHOWNORMAL);            
  }
        free(pszString);
  RegCloseKey(hSubKey);
 }
}

void RemoveChar(LPSTR lpSrc, char chRemove)
{
 LPTSTR pstrSource = lpSrc;
 LPTSTR pstrDest = lpSrc;
 LPTSTR pstrEnd = lpSrc + strlen(lpSrc);

 while (pstrSource < pstrEnd)
 {
  if (*pstrSource != chRemove)
  {
   *pstrDest = *pstrSource;
   pstrDest = _tcsinc(pstrDest);
  }
  pstrSource = _tcsinc(pstrSource);
 }
 *pstrDest = '\0';
}

--------------------------------------------------------------------------------
Copyright 1999© 한기용 Last updated: 11/22/2004 12:45:59  Designed By 한기남


37. 윈도우의 최대/최소 크기 설정
윈도우의 캡션을 없앴을 경우 윈도우를 최대화했을 때 아래의 Task Bar가 가려져 버리는 현상이 생기는데.. 캡션이 있으면 Task Bar 위로만 최대화되는데 말입니다. 어떻게 해결할 수 있는 방법이 없나 궁금하네요..

말씀하신 문제를 해결하려면 WM_GETMINMAXINFO 메시지를 처리해야 합니다. 이는 윈도우의 최대 크기 등을 설정하기 위해 사용되는 메시지입니다. 다음과 같이 처리합니다.

case WM_GETMINMAXINFO :
{
    LPMINMAXINFO lpmmi;
    RECT rc;

    SystemParametersInfo(SPI_GETWORKAREA, 0, &rc,0);
    lpmmi = (LPMINMAXINFO)lParam; 
    lpmmi->ptMaxSize.x = rc.right; 
    lpmmi->ptMaxSize.y = rc.bottom; 
    lpmmi->ptMaxPosition.x = 0; 
    lpmmi->ptMaxPosition.y = 0; 
    lpmmi->ptMinTrackSize.x = GetSystemMetrics(SM_CXMINTRACK); 
    lpmmi->ptMinTrackSize.y = GetSystemMetrics(SM_CYMINTRACK); 
    lpmmi->ptMaxTrackSize.x = GetSystemMetrics(SM_CXMAXTRACK); 
    lpmmi->ptMaxTrackSize.y = GetSystemMetrics(SM_CYMAXTRACK); 
    break;
}


38. Thread에서 Automation 메소드 호출시 에러 발생
Thread를 생성하고 Automation 메소드를 호출했는데 에러가 발생합니다.

App 클래스의 InitInstance 함수에서 AfxOleInit를 호출하는 부분을 CoInitializeEx(NULL, COINIT_MULTITHREADED)를 호출하는 것으로 변경하기 바랍니다. 그리고 App 클래스에 ExitInstance 함수를 추가하고 거기서 CoUninitialize를 호출하도록 하면 됩니다. MFC의 AfxOleInit는 기본적으로 STA(Single Threading Apartment) 모델을 사용합니다. Thread에서 자신이 생성하지 않는 COM 객체를 접근할 때는 MTA(Multiple Threading Apartment) 모델을 사용해야 합니다.


39. 최상위 윈도우의 종료 방법
현재 최상위 윈도우를 찾아서 종료하는 코드를 만들고 싶습니다.

일단 현재 사용자가 작업 중인 최상위 윈도우의 핸들은 GetForegroundWindow API로 얻어냅니다. 그런데 그 윈도우가 자식 윈도우일 수 있기 때문에 GetParent API를 반복적으로 사용해서 최상위 탑 레벨 윈도우의 핸들을 알아냅니다. 종료하는 방법은 먼저 DestroyWindow를 호출해서 시도해보고 실패하면 시스템 메뉴의 "닫기" 명령을 이용해 처리합니다. 사실 이 것도 실패할 수 있는데 무조건 종료시키고 싶다면 아래 코드에서 주석 처리해 놓은 GetWindowThreadProcessId/Terminate API 부분을 사용하면 됩니다.

    HWND hTopWnd = GetForegroundWindow();
    if (hTopWnd == NULL)
    {
        return;
    }

    while(GetParent(hTopWnd))
    {
        hTopWnd = GetParent(hTopWnd);
    }

    if (DestroyWindow(hTopWnd) == FALSE)
    {
        SendMessage(hTopWnd, WM_SYSCOMMAND, SC_CLOSE, NULL);
        //GetWindowThreadProcessId(hTopWnd, &dwProcessId);
        //TerminateProcess(dwProcessId);
    }   

        

40. 인터넷 익스플로러의 위치 경로 알아내기
인터넷 익스플로러가 설치된 절대 경로를 알고 싶습니다.

레지스트리의 HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\App Paths\IExplore.exe 의 기본 값으로 인터넷 익스플로러의 설치 경로가 들어옵니다. 다음 함수를 호출하면 설치 경로를 얻어 줍니다. 인자로 넘어가는 lpPath는 적어도 256바이트 이상의 크기를 갖는 문자 배열이어야 합니다.

BOOL GetIEPath(LPTSTR lpPath) 
{
    long lRet;
    HKEY hKey;

    lRet = RegOpenKey(HKEY_LOCAL_MACHINE, "Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\IExplore.exe", &hKey);
    if (lRet == ERROR_SUCCESS)
    {
        long cbData = 255;

        RegQueryValue(hKey, "", lpPath, &cbData);
        RegCloseKey(hKey);
    }
    else
        return FALSE;
    return TRUE;
}

'Programming > Win32 API' 카테고리의 다른 글

[Win32 API] Win32 API, Window와 Unix  (0) 2015.05.11
[Win32 API] 컨트롤도 윈도우다  (0) 2015.05.11
[Win32 API] 동기화 처리방법  (0) 2015.05.11
[Win32 API] 동기화  (0) 2015.05.11
[Win32 API] CreateFile 함수  (0) 2015.05.11
posted by Kanais