场景
- 有时候我们需要单独对某个窗口消息进行拦截,比如CEdit响应回车, 这时候就需要拦截窗口处理过程了. 当然MFC的界面可以重载:
- 也比如CEdit设置ES_MULTILINE属性后并不支持ctrl+a全选, 这时候也需要自己过滤消息增加全选.
BOOL CXXXDlg::PreTranslateMessage(MSG* pMsg){
但是WTL的CEdit并不支持这种方式,WTL如果想在 PreTranslateMessage 里拦截消息,必须继承 CMessageFilter 后还要把这个控件注册到消息循环里才行,也就是必须写子类 或者从父窗口拦截这个CEdit的消息.
CMessageLoop* pLoop = _Module.GetMessageLoop();
ATLASSERT(pLoop != NULL);
pLoop->AddMessageFilter(this);
方案
- 通过使用 SetWindowLong来改变窗口处理过程来处理相关的消息,其他消息使用原过程继续处理.
static WNDPROC OldWndProc = NULL;
static UiPreviewListDialog* gDialog = NULL;
static LRESULT CALLBACK NewEditProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) // 该对话框新的窗口回调函数,过滤WM_KEYDOWN消息。
{
switch(message)
{
case WM_GETDLGCODE:
{
return (DLGC_WANTALLKEYS | CallWindowProc(OldWndProc, hWnd, message, wParam, lParam)); // 注意这里,否则没有办法捕获Tab/方向键
}
case WM_KEYDOWN:
{
if(wParam == VK_RETURN)
{
std::cout << "Enter: " << std::endl;
BOOL handle;
gDialog->OnSearch(message,wParam,hWnd,handle);
}
}
break;
default:
break;
}
return CallWindowProc(OldWndProc, hWnd, message, wParam, lParam);
}
gDialog = this;
OldWndProc = (WNDPROC)edit_.SetWindowLong(GWL_WNDPROC, (LONG)NewEditProc);
- 全选,使用ES_MULTILINE后默认不能使用ctrl+a全选.
static std::map<HWND,WNDPROC> gProcs;
static LRESULT CALLBACK Edit_Prc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam){
if(msg == WM_CHAR && wParam==1)
{
SendMessage(hwnd,EM_SETSEL,0,-1); return 1;
}
else
{
return CallWindowProc(gProcs[hwnd],hwnd,msg,wParam,lParam);
}
}
WNDPROC p_code = (WNDPROC)edit_code_.SetWindowLong(GWL_WNDPROC, (LONG)Edit_Prc);
WNDPROC p_email = (WNDPROC)edit_email_.SetWindowLong(GWL_WNDPROC, (LONG)Edit_Prc);
gProcs[edit_code_.m_hWnd] = p_code;
gProcs[edit_email_.m_hWnd] = p_email;
参考:
CEdit中对于回车键的响应
SetWindowLong function
CallWindowProc
select-all-text-in-edit-contol-by-clicking-ctrla