【发布时间】:2013-07-23 14:08:44
【问题描述】:
我知道在按下 Enter 或 Esc 键时防止 MFC 对话框关闭的一种方法,但我想了解更多有关该过程的详细信息和这样做的所有常见替代方法。
提前感谢您的帮助。
【问题讨论】:
-
鉴于有两个投票率很高的答案似乎解决了这个问题,我不确定这是什么被关闭为不清楚。
-
对于那些没有线索的人来说,以投票方式结束不清楚的问题,这方式太容易了。
我知道在按下 Enter 或 Esc 键时防止 MFC 对话框关闭的一种方法,但我想了解更多有关该过程的详细信息和这样做的所有常见替代方法。
提前感谢您的帮助。
【问题讨论】:
在处理 Dialog 风格的 MFC 应用程序时,框架会自动对一些必须重写的项目进行硬编码,以防止应用程序在 Esc 或 Enter 键按下时退出按下。但是有一种非常简单的方法,不需要任何特殊的东西,例如实现 PreTranslateMessage(),这是非常不推荐的。
需要具备三个功能:
在标题中,添加三个函数原型。 如果您想添加 WM_CLOSE 事件处理程序,您可以使用类向导,但只需键入它就非常简单。
// DefaultDialogAppDlg.h
//
class CDefaultDialogAppDlg : public CDialogEx
{
// ... other code
protected:
virtual void OnCancel(){} // inline empty function
virtual void OnOK(){} // inline empty function
public:
afx_msg void OnClose(); // message handler for WM_CLOSE
// ...other code
};
在 .cpp 文件中,将 ON_WM_CLOSE() 条目添加到消息映射和三个函数的定义中。由于 OnCancel() 和 OnOK() 通常为空,因此您可以根据需要将它们内联在标题中(请参阅我在第 1 步中所做的操作?)。
.cpp 文件将具有以下内容:
// DefaultDialogAppDlg.cpp
// ... other code
BEGIN_MESSAGE_MAP(CDefaultDialogAppDlg, CDialogEx)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_WM_CLOSE() // WM_CLOSE messages are handled here.
END_MESSAGE_MAP()
// ... other code
void CDefaultDialogAppDlg::OnClose()
{
// TODO: Add exit handling code here
// NOTE: to actually allow the program to end, call the base class
// version of either the OnOK() or OnCancel() function.
//CDialogEx::OnOK(); // returns 1 to theApp object
CDialogEx::OnCancel(); // returns 2 to theApp object
}
【讨论】:
@the-forest-and-the-trees 的回答非常好。除了@oneworld 解决的一种情况。您需要过滤不属于对话窗口的消息:
BOOL CDialogDemoDlg::PreTranslateMessage(MSG* pMsg)
{
if (pMsg->hwnd == this->m_hWnd && pMsg->message == WM_KEYDOWN)
{
if (pMsg->wParam == VK_RETURN || pMsg->wParam == VK_ESCAPE)
{
return TRUE; // Do not process further
}
}
return CWnd::PreTranslateMessage(pMsg);
}
记得在头文件中添加virtual。
【讨论】:
确保您没有#define CUSTOM_ID 2,因为2 已经定义为转义,我认为1 定义为输入?如果我错了,请纠正我。
【讨论】:
我只是重写 OnOk 事件,而不是将消息传递给父对话框,什么都不做。
所以这样做基本上很简单:
void OnOk() override { /*CDialog::OnOK();*/ }
这应该可以防止对话框在按下回车键时关闭。
【讨论】:
有一个替代上一个答案的方法,如果您希望仍然有一个确定/关闭按钮,这很有用。如果你重写 PreTranslateMessage 函数,你可以像这样捕获 VK_ESCAPE / VK_RETURN 的使用:
BOOL MyCtrl::PreTranslateMessage(MSG* pMsg)
{
if( pMsg->message == WM_KEYDOWN )
{
if(pMsg->wParam == VK_RETURN || pMsg->wParam == VK_ESCAPE)
{
return TRUE; // Do not process further
}
}
return CWnd::PreTranslateMessage(pMsg);
}
【讨论】:
WM_GETDLGCODE 可能会更好。
当用户在对话框中按下 Enter 键时,可能会发生两种情况:
CDialog::SetDefID())。然后将带有此控件 ID 的 WM_COMMAND 发送到对话框。使用第一个选项,默认控件的 ID 可能等于 IDOK。那么结果将与第二个选项中的结果相同。
默认情况下,CDialog 类有一个用于调用CDialog::OnOk() 的WM_COMMAND(IDOK) 的处理程序,这是一个虚函数,默认情况下它调用关闭对话框的EndDialog(IDOK)。
因此,如果您想避免关闭对话框,请执行以下操作之一。
IDOK 以外的其他控件。WM_COMMAND(IDOK) 设置一个不调用EndDialog() 的处理程序。CDialog::OnOk() 并且不调用基本实现。关于 IDCANCEL,它是类似的,但没有等效的 SetDefID() 并且 ESC 键是硬编码的。所以为了避免对话框被关闭:
WM_COMMAND(IDCANCEL) 设置一个不调用EndDialog() 的处理程序。CDialog::OnCancel() 并且不调用基本实现。【讨论】:
WM_CLOSE,默认情况下会转换为WM_COMMAND(IDCANCEL)。您可以通过覆盖 OnCancel() 来执行您想要的操作,添加一个直接执行 EndDialog(IDCANCEL) 的 OnClose()。