Microsoft MVP성태의 닷넷 이야기
Windows: 249. Win32 C/C++ - 대화창 템플릿을 런타임에 코딩해서 사용 [링크 복사], [링크+제목 복사],
조회: 10902
글쓴 사람
정성태 (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




Win32 C/C++ - 대화창 템플릿을 런타임에 코딩해서 사용

지난 글에서,

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

rc 파일로 컴파일된 대화창 Template의 2진 코드를 해석하는 방법을 알아봤는데요, 그렇다면 거꾸로 적절한 2진 코드만 만들 수 있다면 대화창이 되는 것과 다를 바가 없습니다. 뭐랄까, 어셈블리 코드만 맞춰주면 함수로 호출할 수 있는 것과 유사한데요, 바로 아래의 글에서,

Building a dialog template at run-time
; https://devblogs.microsoft.com/oldnewthing/20050429-00/?p=35743

그 방법을 설명하고 있습니다. 우선, 아래와 같이 바이너리 버퍼를 쉽게 다룰 클래스를 하나 만들고,

// https://devblogs.microsoft.com/oldnewthing/20050429-00/?p=35743
class DialogTemplate {
public:
    LPCDLGTEMPLATE Template() { return (LPCDLGTEMPLATE)&v[0]; }
    void AlignToDword()
    {
        if (v.size() % 4) Write(NULL, 4 - (v.size() % 4));
    }
    void Write(LPCVOID pvWrite, DWORD cbWrite) {
        v.insert(v.end(), cbWrite, 0);
        if (pvWrite) CopyMemory(&v[v.size() - cbWrite], pvWrite, cbWrite);
    }
    template<typename T> void Write(T t) { Write(&t, sizeof(T)); }
    void WriteString(LPCWSTR psz)
    {
        Write(psz, (lstrlenW(psz) + 1) * sizeof(WCHAR));
    }
private:
    vector<BYTE> v;
};

이후 "Win32 - 리소스에 포함된 대화창 Template의 2진 코드 해석 방법" 글에서 설명한 포맷에 따라 2진 코드를 써주면,

BOOL FakeMessageBox(HWND hwnd, LPCWSTR pszMessage, LPCWSTR pszTitle)
{
    BOOL fSuccess = FALSE;
    HDC hdc = GetDC(NULL);
    if (hdc) {
        NONCLIENTMETRICSW ncm = { sizeof(ncm) };
        if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, 0, &ncm, 0)) {
            DialogTemplate tmp;
            // Write out the extended dialog template header
            tmp.Write<WORD>(1); // dialog version
            tmp.Write<WORD>(0xFFFF); // extended dialog template
            tmp.Write<DWORD>(0); // help ID
            tmp.Write<DWORD>(0); // extended style
            tmp.Write<DWORD>(WS_CAPTION | WS_SYSMENU | DS_SETFONT | DS_MODALFRAME);
            tmp.Write<WORD>(2); // number of controls
            tmp.Write<WORD>(32); // X
            tmp.Write<WORD>(32); // Y
            tmp.Write<WORD>(200); // width
            tmp.Write<WORD>(80); // height
            tmp.WriteString(L""); // no menu
            tmp.WriteString(L""); // default dialog class
            tmp.WriteString(pszTitle); // title
            // Next comes the font description.
            // See text for discussion of fancy formula.
            // 참고: Win32 - 대화창의 DLU 단위를 pixel로 변경하는 방법
            if (ncm.lfMessageFont.lfHeight < 0) {
                ncm.lfMessageFont.lfHeight = -MulDiv(ncm.lfMessageFont.lfHeight,
                    72, GetDeviceCaps(hdc, LOGPIXELSY));
            }
            tmp.Write<WORD>((WORD)ncm.lfMessageFont.lfHeight); // point
            tmp.Write<WORD>((WORD)ncm.lfMessageFont.lfWeight); // weight
            tmp.Write<BYTE>(ncm.lfMessageFont.lfItalic); // Italic
            tmp.Write<BYTE>(ncm.lfMessageFont.lfCharSet); // CharSet
            tmp.WriteString(ncm.lfMessageFont.lfFaceName);
            // Then come the two controls.  First is the static text.
            tmp.AlignToDword();
            tmp.Write<DWORD>(0); // help id
            tmp.Write<DWORD>(0); // window extended style
            tmp.Write<DWORD>(WS_CHILD | WS_VISIBLE); // style
            tmp.Write<WORD>(7); // x
            tmp.Write<WORD>(7); // y
            tmp.Write<WORD>(200 - 14); // width
            tmp.Write<WORD>(80 - 7 - 14 - 7); // height
            tmp.Write<DWORD>(-1); // control ID
            tmp.Write<DWORD>(0x0082FFFF); // static
            tmp.WriteString(pszMessage); // text
            tmp.Write<WORD>(0); // no extra data
            // Second control is the OK button.
            tmp.AlignToDword();
            tmp.Write<DWORD>(0); // help id
            tmp.Write<DWORD>(0); // window extended style
            tmp.Write<DWORD>(WS_CHILD | WS_VISIBLE |
                WS_GROUP | WS_TABSTOP | BS_DEFPUSHBUTTON); // style
            tmp.Write<WORD>(75); // x
            tmp.Write<WORD>(80 - 7 - 14); // y
            tmp.Write<WORD>(50); // width
            tmp.Write<WORD>(14); // height
            tmp.Write<DWORD>(IDCANCEL); // control ID
            tmp.Write<DWORD>(0x0080FFFF); // static
            tmp.WriteString(L"OK"); // text
            tmp.Write<WORD>(0); // no extra data
            // Template is ready - go display it.
            fSuccess = DialogBoxIndirect(hInst, tmp.Template(),
                hwnd, AboutModal) >= 0;
        }
        ReleaseDC(NULL, hdc); // fixed 11 May
    }
    return fSuccess;
}

실제로 이 코드를 다음과 같이 실행해 주면,

case WM_CHAR:
    if (wParam == ' ')
    {
        FakeMessageBox(hWnd,
            L"This is the text of a dynamically-generated dialog template. "
            L"If Raymond had more time, this dialog would have looked prettier.",
            L"Title of message box");
    }
    break;

이렇게 대화창이 뜨는 것을 볼 수 있습니다.

runtime_dialog_template_1.png

물론, 그동안 만들었던,

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

DLGTEMPLATEEX::ReadDialog를 이용해 동일한 template 바이너리를 읽어 대화창을 보여줄 수 있습니다.

BOOL FakeMessageBox(HWND hwnd, LPCWSTR pszMessage, LPCWSTR pszTitle)
{
    BOOL fSuccess = FALSE;
    HDC hdc = GetDC(NULL);
    if (hdc) {
        NONCLIENTMETRICSW ncm = { sizeof(ncm) };
        if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, 0, &ncm, 0)) {
            DialogTemplate tmp;
            // Write out the extended dialog template header
            ...[생략]...
            
            // fSuccess = DialogBoxIndirect(hInst, tmp.Template(), hwnd, AboutModal) >= 0;

            ShowMyDialogWindow(hwnd, (LPVOID)tmp.Template(), AboutModeless);

        }
        ReleaseDC(NULL, hdc); // fixed 11 May
    }
    return fSuccess;
}

void ShowMyDialogWindow(HWND hWnd, LPVOID lpv, DLGPROC dlgProc)
{
    // EndDialog는 메시지 루프를 빠져나오지 않는 문제가 있는데
    // 이를 해결하기 위해서는 별도의 EndManualModalDialog 식의 사용자 정의 종료 함수를 만들어야 함!
    // DLGTEMPLATEEX* item = DLGTEMPLATEEX::ReadDialog(hInst, hWnd, AboutModal, lpv);

    DLGTEMPLATEEX* item = DLGTEMPLATEEX::ReadDialog(hInst, hWnd, dlgProc, lpv);

    item->ApplyDsStyle();
    item->LoadMenu();
    item->LoadFont();
    item->DLUtoPixel();
    if (item->Create() == FALSE)
    {
        delete item;
    }

    item->DoModal();
}

void ShowMyDialogWindow(HWND hWnd)
{
    HRSRC hrsrc = ::FindResource(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), RT_DIALOG);
    if (hrsrc == nullptr)
    {
        return;
    }

    HGLOBAL hglobal = ::LoadResource(hInst, hrsrc);
    if (hglobal == nullptr)
    {
        return;
    }

    LPVOID lpv = ::LockResource(hglobal);
    DWORD resSize = SizeofResource(hInst, hrsrc);

    ShowMyDialogWindow(hWnd, lpv, AboutModeless);

    ::FreeResource(hglobal);
}

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




마지막으로 RC Compiler와 C Compiler가 유사하면서도 다른 2가지 흥미 있는 이야기가 있으니 시간 나시면 한번 읽어보세요. ^^

The Resource Compiler’s preprocessor is not the same as the C preprocessor
; https://devblogs.microsoft.com/oldnewthing/20171004-00/?p=97126

What is the expression language used by the Resource Compiler for non-preprocessor expressions?
; https://devblogs.microsoft.com/oldnewthing/20230313-00/?p=107928




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







[최초 등록일: ]
[최종 수정일: 4/28/2023]

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

비밀번호

댓글 작성자
 




... 46  47  48  49  [50]  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12691정성태6/30/202115724개발 환경 구성: 573. 테스트 용도이지만 테스트에 적합하지 않은 Azure D1 공유(shared) 요금제
12690정성태6/28/202116753Java: 23. Azure - 자바(Java)로 만드는 Web App Service - Tomcat 호스팅
12689정성태6/25/202118432오류 유형: 730. Windows Forms 디자이너 - The class Form1 can be designed, but is not the first class in the file. [1]
12688정성태6/24/202117753.NET Framework: 1073. C# - JSON 역/직렬화 시 리플렉션 손실을 없애는 JsonSrcGen [2]파일 다운로드1
12687정성태6/22/202115089오류 유형: 729. Invalid data: Invalid artifact, java se app service only supports .jar artifact
12686정성태6/21/202117095Java: 22. Azure - 자바(Java)로 만드는 Web App Service - Java SE (Embedded Web Server) 호스팅
12685정성태6/21/202118326Java: 21. Azure Web App Service에 배포된 Java 프로세스의 메모리 및 힙(Heap) 덤프 뜨는 방법
12684정성태6/19/202116669오류 유형: 728. Visual Studio 2022부터 DTE.get_Properties 속성 접근 시 System.MissingMethodException 예외 발생
12683정성태6/18/202117946VS.NET IDE: 166. Visual Studio 2022 - Windows Forms 프로젝트의 x86 DLL 컨트롤이 Designer에서 오류가 발생하는 문제 [1]파일 다운로드1
12682정성태6/18/202114492VS.NET IDE: 165. Visual Studio 2022를 위한 Extension 마이그레이션
12681정성태6/18/202114791오류 유형: 727. .NET 2.0 ~ 3.5 + x64 환경에서 System.EnterpriseServices 참조 시 CS8012 경고
12680정성태6/18/202116865오류 유형: 726. python2.7.exe 실행 시 0xc000007b 오류
12679정성태6/18/202116938COM 개체 관련: 23. CoInitializeSecurity의 전역 설정을 재정의하는 CoSetProxyBlanket 함수 사용법파일 다운로드1
12678정성태6/17/202115438.NET Framework: 1072. C# - CoCreateInstance 관련 Inteop 오류 정리파일 다운로드1
12677정성태6/17/202118280VC++: 144. 역공학을 통한 lxssmanager.dll의 ILxssSession 사용법 분석파일 다운로드1
12676정성태6/16/202117417VC++: 143. ionescu007/lxss github repo에 공개된 lxssmanager.dll의 CLSID_LxssUserSession/IID_ILxssSession 사용법파일 다운로드1
12675정성태6/16/202115357Java: 20. maven package 명령어 결과물로 (war가 아닌) jar 생성 방법
12674정성태6/15/202116531VC++: 142. DEFINE_GUID 사용법
12673정성태6/15/202117161Java: 19. IntelliJ - 자바(Java)로 만드는 Web App을 Tomcat에서 실행하는 방법
12672정성태6/15/202118846오류 유형: 725. IntelliJ에서 Java webapp 실행 시 "Address localhost:1099 is already in use" 오류
12671정성태6/15/202127562오류 유형: 724. Tomcat 실행 시 Failed to initialize connector [Connector[HTTP/1.1-8080]] 오류
12670정성태6/13/202117493.NET Framework: 1071. DLL Surrogate를 이용한 Out-of-process COM 개체에서의 CoInitializeSecurity 문제파일 다운로드1
12669정성태6/11/202117601.NET Framework: 1070. 사용자 정의 GetHashCode 메서드 구현은 C# 9.0의 record 또는 리팩터링에 맡기세요.
12668정성태6/11/202120113.NET Framework: 1069. C# - DLL Surrogate를 이용한 Out-of-process COM 개체 제작파일 다운로드2
12667정성태6/10/202117995.NET Framework: 1068. COM+ 서버 응용 프로그램을 이용해 CoInitializeSecurity 제약 해결파일 다운로드1
12666정성태6/10/202115593.NET Framework: 1067. 별도 DLL에 포함된 타입을 STAThread Main 메서드에서 사용하는 경우 CoInitializeSecurity 자동 호출파일 다운로드1
... 46  47  48  49  [50]  51  52  53  54  55  56  57  58  59  60  ...