如何确定当用户单击""Cancel""时 FindText 对话框已关闭



这段代码产生简单的FindText对话框窗口,当用户单击右上方的X窗口关闭按钮时,WM_CLOSE消息被发送到钩子子程,但是当单击"cancel"按钮时,没有消息显示窗口停止。

#include <windows.h>
#include <iostream>
#include <iomanip>
UINT_PTR CALLBACK FRHookProc(HWND hdlg, UINT uiMsg, WPARAM /*wParam*/, LPARAM /*lParam*/) {
  switch (uiMsg) {
    case WM_INITDIALOG: {
      ShowWindow(hdlg, SW_SHOW);
      break;
    }
  }
  using namespace std;
  if (uiMsg == WM_CLOSE) cout << "FindTextW window has been closed";
  return 0;
}
LRESULT CALLBACK MyWndProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
{
  return DefWindowProcW(hWnd, Msg, wParam, lParam);
}
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE, LPTSTR, int /* nCmdShow*/) {
  using namespace std;
  WNDCLASSEXW wc;
  wc.cbSize = sizeof(WNDCLASSEXW);
  wc.style = CS_HREDRAW | CS_VREDRAW;
  wc.lpfnWndProc = &MyWndProc;
  wc.cbClsExtra = 0;
  wc.cbWndExtra = sizeof(PVOID);
  wc.hInstance = hInstance;
  wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  wc.hIconSm = NULL;
  wc.hCursor = LoadCursor(NULL, IDC_ARROW);
  wc.hbrBackground = reinterpret_cast<HBRUSH>(COLOR_BACKGROUND);
  wc.lpszMenuName = L"MainMenu";
  wc.lpszClassName = L"window";
  ATOM class_atom = RegisterClassExW(&wc);
  HWND hWnd = CreateWindowExW(
    0,
    reinterpret_cast<LPCWSTR>(class_atom),
    L"window title",
    WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPCHILDREN | WS_THICKFRAME,
    CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
    NULL,
    NULL,
    hInstance,
    NULL
  );
  WCHAR szFindWhat[MAX_PATH] = {0};
  FINDREPLACEW fr;
  ZeroMemory(&fr, sizeof(fr));
  fr.lStructSize = sizeof(fr);
  fr.hwndOwner = hWnd;
  fr.lpstrFindWhat = szFindWhat;
  fr.lpfnHook = FRHookProc;
  fr.wFindWhatLen = MAX_PATH;
  fr.Flags = FR_ENABLEHOOK;
  /*HWND hdlg =*/ FindTextW(&fr);
  MSG msg;
  for (;;) {
    GetMessageW(&msg, 0, 0, 0);
    TranslateMessage(&msg);
    DispatchMessageW(&msg);
  }
  return 0;
}

请仔细阅读文档。您采用了完全错误的方法来检测对话框窗口是否关闭。

FindText功能:

在调用FindText之前,必须调用RegisterWindowMessage函数来获取FINDMSGSTRING消息的标识符。当用户单击"查找下一个"按钮时,或当对话框关闭时,对话框过程使用此标识符发送消息。

FINDMSGSTRING信息:

查找或替换对话框发送FINDMSGSTRING注册消息到其所有者窗口的窗口过程,当用户单击查找下一个,替换或替换所有按钮时,或关闭对话框

必须在调用RegisterWindowMessage函数时指定FINDMSGSTRING常量,以获得对话框发送的消息的标识符。

创建对话框时,使用FINDREPLACE结构的hwndOwner成员来标识接收FINDMSGSTRING消息的窗口

FINDREPLACE结构的Flags成员包括以下标志之一,以指示引起该消息的事件。

FR_DIALOGTERM (0 x00000040)对话框正在关闭。在所有者窗口处理此消息后,对话框的句柄不再有效。

所以,说了这么多,试试这个:

#include <windows.h>
#include <iostream>
#include <iomanip>
static const UINT uFindMsgString = RegisterWindowMessageW(L"commdlg_FindReplace");
LRESULT CALLBACK MyWndProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
{
  if ((Msg == uFindMsgString) && (Msg != 0))
  {
    FINDREPLACE *fr = reinterpret_cast<FINDREPLACE*>(lParam);
    if (fr->flags & FR_DIALOGTERM)
    {
      std::cout << "FindTextW window has been closed";
      PostQuitMessage(0);
    }
    // process other dialog notifications as needed...
    return 0;
  }
  return DefWindowProcW(hWnd, Msg, wParam, lParam);
}
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE, LPTSTR, int /* nCmdShow*/)
{
  WNDCLASSEXW wc = {0};
  wc.cbSize = sizeof(WNDCLASSEXW);
  wc.style = CS_HREDRAW | CS_VREDRAW;
  wc.lpfnWndProc = &MyWndProc;
  wc.cbClsExtra = 0;
  wc.cbWndExtra = sizeof(PVOID);
  wc.hInstance = hInstance;
  wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  wc.hIconSm = NULL;
  wc.hCursor = LoadCursor(NULL, IDC_ARROW);
  wc.hbrBackground = reinterpret_cast<HBRUSH>(COLOR_BACKGROUND);
  wc.lpszMenuName = L"MainMenu";
  wc.lpszClassName = L"window";
  ATOM class_atom = RegisterClassExW(&wc);
  HWND hWnd = CreateWindowExW(
    0,
    wc.lpszClassName,
    L"window title",
    WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPCHILDREN | WS_THICKFRAME,
    CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
    NULL,
    NULL,
    hInstance,
    NULL
  );
  WCHAR szFindWhat[MAX_PATH] = {0};
  FINDREPLACEW fr = {0};
  fr.lStructSize = sizeof(fr);
  fr.hwndOwner = hWnd;
  fr.lpstrFindWhat = szFindWhat;
  fr.wFindWhatLen = MAX_PATH;
  HWND hdlg = FindTextW(&fr);
  MSG msg;
  while (GetMessageW(&msg, 0, 0, 0) > 0)
  {
    if (!IsDialogMessage(hdlg, &msg))
    {
      TranslateMessage(&msg);
      DispatchMessageW(&msg);
    }
  }
  DestroyWindow(hWnd);
  UnregisterClass(wc.lpszClassName, hInstance);
  return 0;
}

您应该收到取消按钮的BN_CLICKED消息的WM_COMMAND通知。当'cancel'被按下时,你应该得到IDCANCEL控件ID的通知。

相关内容

  • 没有找到相关文章

最新更新