一、菜单与工具按钮同步响应的问题
要使菜单、工具栏按钮响应同一个事件只需要将两者ID号设置为相同即可;我在这里主要说明怎样使两者的ON_UPDATE_COMMAND_UI命令得到一致的响应:
(1)无论是模态对话框还是非模态对话框,要更新菜单与工具按钮的状态,即添ON_UPDATE_COMMAND_UI消息响应。但是,结果出乎我们意料,对话并没有任何响应:
从命令用户界面处理函数(Command UI handler)改变菜单状态(启用/禁用,选择/取消选择,更改文字)在由对话框处理时没有正常工作。
void CTestDlg::OnUpdate**********(CCmdUI*pCmdUI)
{
pCmdUI->SetCheck(TRUE); // 没有文字前显示选定标记.
pCmdUI->SetRadio(TRUE); // 没有在文字前显示点.
pCmdUI->SetText("Close"); //没有更改菜单文字.
}
原因:在下拉菜单显示的时候, WM_INITMENUPOPUP消息被先发送以显示菜单项。MFC CFrameWnd::OnInitMenuPopup函数遍历菜单项并为每个菜单项调用更新命令处理函数(如果有的话),.菜单的外观被更新以反映它的状态(启用/禁用,选择/取消选择),更新用户界面机制在基于对话框的应用程序中不能工作,因为CDialog没有OnInitMenuPopup处理函数,而使用CWnd's默认处理函数,该函数没有为菜单项调用更新命令处理函数。
解决办法:添加该函数。可以使用类向导来添加(即:项目->类向导->类名选”CXXXDlg”->消息里搜” WM_INITMENUPOPUP”->添加)如果消息里没有该消息,可以使用类向导类的”添加自定义消息”进行添加或进行手动添加。
具体细节见:http://blog.csdn.net/johnnyhu90/article/details/9623207
在此主要补充http://blog.csdn.net/johnnyhu90/article/details/9623207中所说在非模态对话框中的不同。
对于非模态对话框,它没有自己单独的消息循环,工具条总是无法响应ON_UPDATE_COMMAND_UI消息;所以我的解决
办法是:
在工具按钮对应的消息响应函数汇总手动添加更改工具按钮状态的代码:m_ToolBar.GetToolBarCtrl().CheckButton(nID,bcheck);以此来更新工具按钮的状态。
二、工具按钮提示信息
“工具按钮的提示信息”就是希望我们鼠标落在工具条按钮上时,能够自动的给出按钮的功能简介:
(1)对于模态对话框,也需要手动添加消息映射。网上已有成功的解决方案:http://blog.163.com/zero_li/blog/static/1727255720110130508138/(2)对于非模态对话框,我们发现应用程序无法响应上面手动添加的消息响应,这里给出一种解决方案:添加一个HOOK(钩子)。1.定义一个全局变量:HHOOK hMeasureHook;2.头文件添加函数声明:static LRESULT CALLBACK GetMessageProc(int nCode, WPARAM wParam, LPARAM lParam);3.源文件添加定义:LRESULT CALLBACK CtestDialog::GetMessageProc(int nCode, WPARAM wParam, LPARAM lParam)
{
//AFX_MANAGE_STATE(AfxGetStaticModuleState( ));
LPMSG lpMsg = (LPMSG) lParam;
if( (nCode >= 0) )
{
CWnd *pWnd = CWnd::FromHandlePermanent(lpMsg->hwnd);
if (pWnd) // a CWnd exists
pWnd->PreTranslateMessage(lpMsg);
}
return ::CallNextHookEx(hMeasureHook, nCode, wParam, lParam);
}
4.在OnInitDialog()中添加钩子:hMeasureHook = SetWindowsHookEx(
WH_GETMESSAGE,
GetMessageProc,
NULL,
GetCurrentThreadId());
ASSERT (hMeasureHook);
这样就OK了