0

更新:根据要求,我添加了用于创建 Window 及其 RichEdit 控件的所有代码。

我正在尝试处理用作另一个窗口子级的 RichEdit 控件的 Windows 消息。

现在我确实让 RichEdit 控件与我自己的WndProc. 问题是,当我设置wc.lpszClassName = MSFTEDIT_CLASS;为匹配lpClassNameused inCreateWindowEx()时,RichEdit 控件的内容不再显示为绘制(即文本等),但是,它的 WndProc 函数可以处理消息。

窗口的创建:

首先是构造函数:

SubWindow::SubWindow(const wchar_t *szAppNameImport)
{
    szAppName = szAppNameImport;

    cfmt = CHARFORMATW();
    hwnd = HWND();
    windowRect = RECT();
    editControlHwnd = HWND();
    wc = WNDCLASSEX();

    wc.cbSize = sizeof(WNDCLASSEX);
    wc.style = CS_CLASSDC;
    wc.lpfnWndProc = WndProc;
    wc.cbClsExtra = 0;
    wc.cbWndExtra = 0;
    wc.hInstance = GetModuleHandle(NULL);
    wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
    wc.hCursor = LoadCursor(NULL, IDC_ARROW);
    wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
    wc.lpszMenuName = NULL;
    wc.lpszClassName = szAppName;
    wc.hIconSm = LoadIcon(wc.hInstance, MAKEINTRESOURCE(IDI_APPLICATION));
}

然后是Create()函数:

VOID SubWindow::Create(unsigned int window_startX, unsigned int window_startY, unsigned int windowWidthInput, unsigned int windowHeightInput, HWND parent)
{   
    windowRect.left = window_startX;
    windowRect.top = window_startY;

    windowRect.right = windowWidthInput;
    windowRect.bottom = windowHeightInput;

    if(!RegisterClassEx(&wc))
    {
        throw std::exception();
    }

    if((hwnd = CreateWindowEx
        (
        WS_EX_CLIENTEDGE,
        szAppName,
        TEXT("Our classy sub window!"),
        WS_OVERLAPPEDWINDOW| WS_VISIBLE,

        windowRect.left, windowRect.top,
        windowRect.right, windowRect.bottom,
        parent,
        NULL,       
        wc.hInstance,
        NULL))==NULL)
    {
        throw std::exception();
    }

    SetWindowLongPtr(hwnd, GWL_USERDATA, (LONG_PTR)this);

    ShowWindow(hwnd, SW_SHOWDEFAULT);
    UpdateWindow(hwnd);
}

WndProc:

LRESULT CALLBACK SubWindow::WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    SubWindow *childWindowPointer = (SubWindow*)GetWindowLongPtr(hwnd, GWLP_USERDATA);

    if(childWindowPointer != NULL)
    {
        if(childWindowPointer->GetEditControl() == hwnd)
            OutputDebugString(L"I SHOULD NOT BE CALLED");

        return childWindowPointer->MsgProc(hwnd, uMsg, wParam, lParam);
    }
    else
    {
        return DefWindowProc(hwnd, uMsg, wParam, lParam);
    }
}

消息过程:

LRESULT SubWindow::MsgProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) 
{
    PAINTSTRUCT ps;
    HDC hdc;

    switch(uMsg)
    {
    case WM_WINDOWPOSCHANGED:
        {
            GetClientRect(hwnd, &windowRect);
            SetWindowPos(editControlHwnd, NULL, windowRect.left, windowRect.top, windowRect.right, windowRect.bottom, SWP_NOZORDER | SWP_NOACTIVATE);
            return 0;
        }
    case WM_DESTROY:
        {
            OutputDebugString(TEXT("DESTROYING A SUB WINDOW!\n"));
            return 0;
        }

    case WM_PAINT:
        {
            InvalidateRect (hwnd, NULL, FALSE);
            hdc = BeginPaint(hwnd, &ps);
            EndPaint(hwnd, &ps);
            return 0;
        }

    case EM_EXSETSEL:
        {
            if(hwnd == editControlHwnd)
            {
                OutputDebugString(L"Text selection changed");
                return 0;
            }
        }
    }
    return DefWindowProc(hwnd, uMsg, wParam, lParam);
} 

RichEdit 控件可以完美地绘制和运行,显然没有问题,除了它没有使用WndProc我定义的。

我不确定我在这里做错了什么或如何正确解决这个问题。

编辑: 根据答案和评论,我已恢复我的代码以仅使用Window包含 RichEdit 控件的类,因此创建:

void SubWindow::CreateEditControl()
{
    std::wstring initialText = TEXT("TestWindow\r\n");

    LoadLibrary(L"Msftedit.dll");

    GetClientRect(hwnd, &windowRect);
    editControlHwnd = CreateWindowEx(0, MSFTEDIT_CLASS, initialText.data(),
        WS_CHILD | WS_VISIBLE | ES_MULTILINE | ES_READONLY | WS_VSCROLL | ES_NOHIDESEL,
        windowRect.left, windowRect.top,windowRect.right,windowRect.bottom,
        hwnd,
        NULL, NULL, NULL);

    cfmt.cbSize = sizeof(CHARFORMAT);
    cfmt.dwMask = CFM_COLOR | CFM_FACE | CFM_SIZE;
    cfmt.dwEffects = 0;
    cfmt.yHeight = 160;
    cfmt.crTextColor = RGB(0,0,0);
    wcscpy_s(cfmt.szFaceName, TEXT("Tahoma"));

    SendMessage(editControlHwnd, EM_SETCHARFORMAT, SCF_DEFAULT, (LPARAM)&cfmt);
}

如何在 Window 的 MsgProc 中处理来自此控件的消息?

4

3 回答 3

1

当您使用默认类名 ( ) 创建富编辑控件窗口时MSFTEDIT_CLASS,所有消息都将发送到其父窗口。由于您不是那个父窗口,因此您无法处理这些消息。

因此,您需要对控件进行子类化,替换您自己的将直接调用的窗口过程,而不是允许将消息传递给父级。这很容易做到;我之前在这个答案中讨论过它,用于常规编辑控件。更改后的示例代码如下所示:

// Stores the old original window procedure for the rich edit control.
WNDPROC wpOldRichEditProc;

// The new custom window procedure for the rich edit control.
LRESULT CALLBACK CustomRichEditProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
    switch (msg)
    {
        ...
    }

    // Pass the messages you don't process on to the original window procedure.
    CallWindowProc(wpOldRichEditProc, hWnd, msg, wParam, lParam);
}

当您创建控件时:

// Create the rich edit control
HWND hWnd = CreateWindowEx(...)

// Subclass it.
wpOldRichEditProc= (WNDPROC)SetWindowLongPtr(hWnd,
                                             GWLP_WNDPROC,
                                             (WNDPROC)CustomRichEditProc);

您还需要确保在控件被销毁时取消子类化。另一个示例演示了这样做是为了响应父窗口收到的消息,但这在您的情况下不起作用,因为您没有收到父窗口的消息。相反,您需要从控件中删除子类以响应它自己的WM_NCDESTROY消息:

SetWindowLongPtr(hWnd, GWLP_WNDPROC, (LONG_PTR)wpOldRichEditProc);

或者,通用控件库的版本 6 引入了一种新的、不易出错的使用一组实用函数的子类化方法。(关键功能实际上存在于早期版本中,但没有记录。)考虑到您无法控制实际拥有窗口的进程,这可以说是首选方法。

MSDN 上有这两种方法的演示。

当然,您不必只对单个控件进行子类化。您还可以注册一个自定义窗口类,其行为方式与内置富编辑控件相同,但仍让您首先破解该类窗口接收到的消息。我无法从问题中判断这是否有必要;听起来您只有一个您关心的控件

于 2013-08-24T13:52:00.407 回答
1

您说最初的问题是您的父窗口没有从 RichEdit 控件获取通知消息。您是否向 RichEdit 控件发送了 EM_SETEVENTMASK 消息?如果不这样做,RichEdit 控件将不会向其父窗口发送某些通知消息。请参阅EM_SETEVENTMASK 消息

于 2013-08-24T22:12:20.680 回答
0

你能展示你的代码,包括wc窗口的结构和创建吗?我很确定您不希望主窗口与富编辑控件具有相同的类 - 这就是我目前正在阅读的内容。

我什至不知道您为什么要WNDCLASSEX申请富编辑控件。

我的建议是,在创建富编辑控件后简化事情并“子类化”富编辑控件,使用SetWindowLong()withGWL_WNDPROC到您的EditControl::WndProc.

于 2013-08-24T13:26:07.040 回答