Microsoft MVP성태의 닷넷 이야기
Windows: 232. C/C++ - 일반 창에도 사용 가능한 IsDialogMessage [링크 복사], [링크+제목 복사],
조회: 12792
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

(시리즈 글이 21개 있습니다.)
Windows: 226. Win32 C/C++ - Dialog에서 값을 반환하는 방법
; https://www.sysnet.pe.kr/2/0/13284

Windows: 227. Win32 C/C++ - Dialog Procedure를 재정의하는 방법
; https://www.sysnet.pe.kr/2/0/13285

Windows: 228. Win32 - 리소스에 포함된 대화창 Template의 2진 코드 해석 방법
; https://www.sysnet.pe.kr/2/0/13286

Windows: 229. Win32 - 대화창 템플릿의 2진 리소스를 읽어들여 윈도우를 직접 띄우는 방법
; https://www.sysnet.pe.kr/2/0/13287

Windows: 230. Win32 - 대화창의 DLU 단위를 pixel로 변경하는 방법
; https://www.sysnet.pe.kr/2/0/13288

Windows: 231. Win32 - 대화창 템플릿의 2진 리소스를 읽어들여 자식 윈도우를 생성하는 방법
; https://www.sysnet.pe.kr/2/0/13289

Windows: 232. C/C++ - 일반 창에도 사용 가능한 IsDialogMessage
; https://www.sysnet.pe.kr/2/0/13292

Windows: 233.  Win32 - modeless 대화창을 modal처럼 동작하게 만드는 방법
; https://www.sysnet.pe.kr/2/0/13295

Windows: 234. IsDialogMessage와 협업하는 WM_GETDLGCODE Win32 메시지
; https://www.sysnet.pe.kr/2/0/13296

Windows: 235. Win32 - Code Modal과 UI Modal
; https://www.sysnet.pe.kr/2/0/13297

Windows: 237. Win32 - 모든 메시지 루프를 탈출하는 WM_QUIT 메시지
; https://www.sysnet.pe.kr/2/0/13299

Windows: 238. Win32 - Modal UI 창에 올바른 Owner(HWND)를 설정해야 하는 이유
; https://www.sysnet.pe.kr/2/0/13300

Windows: 242. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (쉬운 버전)
; https://www.sysnet.pe.kr/2/0/13305

Windows: 243. Win32 - 윈도우(cbWndExtra) 및 윈도우 클래스(cbClsExtra) 저장소 사용 방법
; https://www.sysnet.pe.kr/2/0/13306

Windows: 244. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (개선된 버전)
; https://www.sysnet.pe.kr/2/0/13312

Windows: 245. Win32 - 시간 만료를 갖는 컨텍스트 메뉴와 윈도우 메시지의 영역별 정의
; https://www.sysnet.pe.kr/2/0/13315

Windows: 246. Win32 C/C++ - 직접 띄운 대화창 템플릿을 위한 Modal 메시지 루프 생성
; https://www.sysnet.pe.kr/2/0/13329

Windows: 247. Win32 C/C++ - CS_GLOBALCLASS 설명
; https://www.sysnet.pe.kr/2/0/13330

Windows: 248. Win32 C/C++ - 대화창을 위한 메시지 루프 사용자 정의
; https://www.sysnet.pe.kr/2/0/13332

Windows: 249. Win32 C/C++ - 대화창 템플릿을 런타임에 코딩해서 사용
; https://www.sysnet.pe.kr/2/0/13333

Windows: 250. Win32 C/C++ - Modal 메시지 루프 내에서 SetWindowsHookEx를 이용한 Thread 메시지 처리 방법
; https://www.sysnet.pe.kr/2/0/13334




C/C++ - 일반 창에도 사용 가능한 IsDialogMessage

지난 글에서 가볍게 IsDialogMessage API를 언급하면서 끝냈는데요,

C# - 윈도우에서 기본 제공하는 FindText 대화창 사용법
; https://www.sysnet.pe.kr/2/0/13290#winform_isdlgmsg

그렇다면 IsDialogMessage를 언제 사용하는지, 간단하게 예를 들어볼까요? ^^

이를 위해, 우선 비주얼 스튜디오로 만든 C++ Windows Application 프로젝트 상태에서, 기본 RC 파일에 포함된 대화창을 다음과 같이 변경합니다.

IDD_ABOUTBOX DIALOGEX 0, 0, 170, 62
STYLE DS_SETFONT | DS_MODALFRAME | DS_FIXEDSYS | WS_POPUP | WS_CAPTION | WS_SYSMENU
CAPTION "About Project0"
FONT 8, "MS Shell Dlg", 0, 0, 0x1
BEGIN
    ICON            IDR_MAINFRAME,IDC_STATIC,14,14,20,20
    EDITTEXT        IDC_EDIT1,38,7,114,15
    EDITTEXT        IDC_EDIT2,38,24,114,15
    DEFPUSHBUTTON   "OK",IDOK,113,41,50,14,WS_GROUP
END

위의 변경을 오류 없이 컴파일하기 위해 resource.h 파일에 IDC_EDIT1, IDC_EDIT2 상수 정의만 추가한 다음 실행, "Help" / "About" 메뉴를 선택하면 다음과 같은 창이 뜨게 될 것입니다.

is_dialog_msg_1.png

DialogBox API를 사용하는 기본 예제 코드의 About 대화 상자는 Modal 형식으로 동작하게 되는데요, "Tab" 키를 누르면 대화상자 내에 있는 2개의 에디트 박스와 버튼 간에 입력 포커스가 이동하는 것을 확인할 수 있습니다.

자, 그럼 응용 프로그램을 종료하고 About 상자를 Modeless로 띄우도록 코드를 일부 변경합니다. (modal을 modeless로 띄우는 방법은 이전에도 한번 예제 코드를 다뤘습니다.)

HWND hwndModeless = nullptr;

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    case WM_COMMAND:
        {
            int wmId = LOWORD(wParam);
            // Parse the menu selections:
            switch (wmId)
            {
            case IDM_ABOUT:
                // DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About); // Modal 창으로 띄우던 코드

                hwndModeless = CreateDialog(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, AboutModeless);
                ShowWindow(hwndModeless, SW_SHOW);
                break;
            case IDM_EXIT:
                DestroyWindow(hWnd);
                break;
            default:
                return DefWindowProc(hWnd, message, wParam, lParam);
            }
        }
        break;
    // ...[생략]...
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

INT_PTR CALLBACK AboutModeless(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
    UNREFERENCED_PARAMETER(lParam);
    switch (message)
    {
    case WM_INITDIALOG:
        return (INT_PTR)TRUE;

    case WM_COMMAND:
        if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
        {
            ::DestroyWindow(hDlg);
            return (INT_PTR)TRUE;
        }
        break;
    }
    return (INT_PTR)FALSE;
}

이제 다시 실행해 볼까요? ^^ 마찬가지로 "Help" / "About" 메뉴를 선택해 띄우면 이제는 해당 대화창이 Modeless로 뜨게 되는데요, 재미있는 건 이번에는 About 대화창에서 Tab 키를 눌러도 입력 포커스가 이동하지 않습니다. 그 외에도 ESC 키를 눌러도 대화창이 종료되지 않고 ENTER 키를 눌러도 DEFPUSHBUTTON으로 지정되어 있던 버튼이 눌리지 않습니다.

그 이유가 바로 "IsDialogMessage" API에 있습니다. 즉, 대화창에서 당연히 누려야 했던 그 기능들을 IsDialogMessage에서 제어해 주고 있던 것입니다. 결국, 위와 같이 Modeless로 뜬 경우에도 대화창의 기존 동작 방식을 허용하고 싶다면 IsDialogMessage를 다음과 같이 Message Loop에 추가만 하면 됩니다.

while (GetMessage(&msg, nullptr, 0, 0))
{
    if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)
        && !IsDialogMessage(hwndModeless, &msg)) // If I have a modeless dialog box with custom accelerators, which should I call first: IsDialogMessage or TranslateAccelerator
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
}

그럼 왜? 이전의 Modal 예제에서는 IsDialogMessage를 호출하지 않은 상태에서도 TAB/ESC 키 등이 먹혔을까요? 왜냐하면 DialogBox API의 경우 블록킹 함수로 동작하는 데, 그 함수 내에서 IsDialogMessage를 호출하는 Message Loop를 실행해 주고 있기 때문입니다.




IsDialogMessage의 역할은 비록 "Dialog"라는 이름을 포함하지만 꼭 대화창에만 쓰는 용도는 아닙니다. 그보다는, 대화창처럼 혜택을 받게 만드는 용도라고 보시면 됩니다. 따라서 일반적인 Window도 TAB/ESC 등의 동작을 IsDialogMessage를 붙이면 가능합니다.

이에 대해서는 다음의 글에서 잘 설명하고 있는데요,

Using the TAB key to navigate in non-dialogs
; https://devblogs.microsoft.com/oldnewthing/20031021-00/?p=42083

실습을 위해, 비주얼 스튜디오에 기본 Windows Application 프로젝트를 생성하고 해당 창에 자식 윈도우를 3개 만들어 둡니다.

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    case WM_CREATE:
        {
            HWND hwndChild =
                CreateWindow(
                    TEXT("button"),                 /* Class Name */
                    TEXT("Button &1"),              /* Title */
                    WS_CHILD | WS_VISIBLE | WS_TABSTOP |
                    BS_DEFPUSHBUTTON | BS_TEXT,     /* Style */
                    0, 0, 100, 100,                 /* Position and size */
                    hWnd,                           /* Parent */
                    (HMENU)100,                     /* Child ID */
                    hInst,                        /* Instance */
                    0);                             /* No special parameters */
            if (!hwndChild) return FALSE;
            g_hwndLastFocus = hwndChild;
            hwndChild =
                CreateWindow(
                    TEXT("button"),                 /* Class Name */
                    TEXT("Button &2"),              /* Title */
                    WS_CHILD | WS_VISIBLE | WS_TABSTOP |
                    BS_PUSHBUTTON | BS_TEXT,        /* Style */
                    100, 0, 100, 100,               /* Position and size */
                    hWnd,                           /* Parent */
                    (HMENU)101,                     /* Child ID */
                    hInst,                        /* Instance */
                    0);                             /* No special parameters */
            if (!hwndChild) return FALSE;
            hwndChild =
                CreateWindow(
                    TEXT("button"),                 /* Class Name */
                    TEXT("Cancel"),                 /* Title */
                    WS_CHILD | WS_VISIBLE | WS_TABSTOP |
                    BS_PUSHBUTTON | BS_TEXT,        /* Style */
                    200, 0, 100, 100,               /* Position and size */
                    hWnd,                           /* Parent */
                    (HMENU)IDCANCEL,                /* Child ID */
                    hInst,                        /* Instance */
                    0);                             /* No special parameters */
            if (!hwndChild) return FALSE;
            return TRUE;
        }
        break;
        
    case WM_SETFOCUS:
        if (g_hwndLastFocus) {
            ::SetFocus(g_hwndLastFocus);
        }
        break;

    case WM_ACTIVATE:
        if (wParam == WA_INACTIVE) {
            g_hwndLastFocus = ::GetFocus();
        }
        break;

    ...[생략]...
        
    case WM_COMMAND:
        {
            int wmId = LOWORD(wParam);
            // Parse the menu selections:
            switch (wmId)
            {
            case 100:
                MessageBox(hWnd, TEXT("Button 1 pushed"), TEXT("Title"), MB_OK);
                break;
            case 101:
                MessageBox(hWnd, TEXT("Button 2 pushed"), TEXT("Title"), MB_OK);
                break;
            case IDCANCEL:
                MessageBox(hWnd, TEXT("Cancel pushed"), TEXT("Title"), MB_OK);
                
            ...[생략]...
            default:
                return DefWindowProc(hWnd, message, wParam, lParam);
            }
        }
        break;

    ...[생략]...

    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

그다음 해야 할 것은, 메시지 루프에 메인 윈도우 핸들을 인자로 받는 IsDialogMessage를 호출하면 끝입니다.

while (GetMessage(&msg, nullptr, 0, 0))
{
    if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg) && !IsDialogMessage(g_hWnd, &msg))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
}

이후 실행하면, 메인 윈도우 내에 생성한 3개의 버튼 윈도우를 TAB 키와 화살표 키를 이동할 수 있고, ESC 키를 눌러 IDCANCEL 버튼을 누른 효과를 낼 수 있습니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




정리해 보면, IsDialogMessage는 Modeless로 띄운 대화창이 있는 경우, 그 대화창에 부가 기능을 넣을 때 호출해 주면 됩니다. 혹은, 일반적인 윈도우에도 그런 부가 기능을 넣고 싶다면 호출할 수 있습니다. 그 외의 경우라면 굳이 Message Loop에 호출 코드를 넣을 필요가 없습니다.

마지막으로, 왜 IsDialogMessage가 필요한 것인지 아래의 글에서 설명하고 있습니다.

Why do we need IsDialogMessage at all?
; https://devblogs.microsoft.com/oldnewthing/20120416-00/?p=7853

Modeless 대화창의 경우 그 창 자체가 포커스를 가지고 있다면 IsDialogMessage가 하는 역할은 대화창의 Windows Procedure 안에서 다룰 수 있습니다. 하지만 Modeless 대화창이 입력 포커스를 가지고 있지 않다면 문제가 발생하는데요, 일례로 대화창 내의 에디트 컨트롤에 입력 포커스가 있을 때에도 대화창은 그 입력을 가로채 부가적인 일을 할 수는 없습니다. (만약, IsDialogMessage 없이 그게 가능하도록 만들려면 대화창은 자신이 소유한 자식 컨트롤에 대한 모든 Window Procedure를 subclassing 해야만 그나마 비슷하게 흉내라도 낼 수 있을 것입니다.)

따라서 입력 포커스를 가진 컨트롤에 메시지가 전달되기 전, 메시지 루프에서의 IsDialogMessage가 중재 역할을 하는 것입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 3/24/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...
NoWriterDateCnt.TitleFile(s)
1303정성태6/26/201227404개발 환경 구성: 152. sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
1302정성태6/25/201229462개발 환경 구성: 151. Azure 웹 사이트에 사용자 도메인 네임 연결하는 방법
1301정성태6/20/201225766오류 유형: 156. KB2667402 윈도우 업데이트 실패 및 마이크로소프트 Answers 웹 사이트 대응
1300정성태6/20/201231788.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 [1]파일 다운로드2
1299정성태6/18/201232897제니퍼 .NET: 21. 제니퍼 닷넷 - Ninject DI 프레임워크의 성능 분석 [2]파일 다운로드2
1298정성태6/14/201234411VS.NET IDE: 72. Visual Studio에서 pfx 파일로 서명한 경우, 암호는 어디에 저장될까? [2]
1297정성태6/12/201231057VC++: 63. 다른 프로세스에 환경 변수 설정하는 방법파일 다운로드1
1296정성태6/5/201227699.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 [4]파일 다운로드1
1295정성태6/5/201225086.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기파일 다운로드1
1294정성태5/27/201248545.NET Framework: 326. 유니코드와 한글 - 유니코드와 닷넷을 이용한 한글 처리 [7]파일 다운로드2
1293정성태5/24/201229776.NET Framework: 325. System.Drawing.Bitmap 데이터를 Parallel.For로 처리하는 방법 [2]파일 다운로드1
1292정성태5/24/201223755.NET Framework: 324. First-chance exception에 대해 조건에 따라 디버거가 멈추게 할 수는 없을까? [1]파일 다운로드1
1291정성태5/23/201230284VC++: 62. 배열 초기화를 위한 기계어 코드 확인 [2]
1290정성태5/18/201235083.NET Framework: 323. 관리자 권한이 필요한 작업을 COM+에 대행 [7]파일 다운로드1
1289정성태5/17/201239242.NET Framework: 322. regsvcs.exe로 어셈블리 등록 시 시스템 변경 사항 [5]파일 다운로드2
1288정성태5/17/201226467.NET Framework: 321. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (3) - Type Library파일 다운로드1
1287정성태5/17/201229305.NET Framework: 320. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (2) - .NET 4.0 + .NET 2.0 [2]
1286정성태5/17/201238233.NET Framework: 319. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (1) - .NET 2.0 + x86/x64/AnyCPU [5]
1285정성태5/16/201233267.NET Framework: 318. gacutil.exe로 어셈블리 등록 시 시스템 변경 사항파일 다운로드1
1284정성태5/15/201225704오류 유형: 155. Windows Phone 연결 상태에서 DRIVER POWER STATE FAILURE 블루 스크린 뜨는 현상
1283정성태5/12/201233316.NET Framework: 317. C# 관점에서의 Observer 패턴 구현 [1]파일 다운로드1
1282정성태5/12/201226110Phone: 6. Windows Phone 7 Silverlight에서 Google Map 사용하는 방법 [3]파일 다운로드1
1281정성태5/9/201233197.NET Framework: 316. WPF/Silverlight의 그래픽 단위와 Anti-aliasing 처리를 이해하자 [1]파일 다운로드1
1280정성태5/9/201226159오류 유형: 154. Could not load type 'System.ServiceModel.Activation.HttpModule' from assembly 'System.ServiceModel, ...'.
1279정성태5/9/201224919.NET Framework: 315. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 [1]파일 다운로드1
1278정성태5/8/201226151오류 유형: 153. Visual Studio 디버깅 - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...