CSDN - 专家门诊 - VC/MFC 界面问题 回复 | 推荐 | 收藏 | 专题 | 公告 | 管理 | 保存 | 关闭窗口 主 题: 如何在MDI客户区窗口响应鼠标双击消息? 作 者: huyoo (空格键) 等 级: 信 誉 值: 100 所属论坛: VC/MFC 界面 问题点数: 20 回复次数: 15 发表时间: 2004-5-15 4:48:26 大家知道,在PhotoShop中的客户区双击鼠标的话,将会弹出打开文件对话框;按住CTRL并双击鼠标的话,会执行新建命令. 我的目的就是,在MDI窗口的客户区中响应鼠标双击消息. 看了《深入MDI客户窗口编程》(在CSDN中有)之后, 我重载了PreTranslateMessage(MSG* pMsg)函数.使它能够在消息发送到TranslateMessage()和DispatchMessage() 函数以前预先解释消息. 可以重载该函数截获MDI客户窗口WM_LBUTTTONDBLCLK消息,我是这样做的: 1.在BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs) { if( !CMDIFrameWnd::PreCreateWindow(cs) ) return FALSE; return TRUE; } 中添加cs.style|=CS_DBLCLKS;以防MDI窗口不响应双击消息. 2.重载PreTranslateMessage(MSG* pMsg)函数 添加代码如下: BOOL CMainFrame::PreTranslateMessage(MSG* pMsg) { //if(pMsg->hwnd==m_hWndMDIClient && pMsg->message==WM_LBUTTONDOWN) //PostMessage(WM_LBUTTONDOWN); //我在这里注释掉拦截鼠标单击的消息,是因为客户区响应了鼠标单击的消息 //而没有响应鼠标双击的消息 if(pMsg->hwnd==m_hWndMDIClient && pMsg->message==WM_LBUTTONDBLCLK) PostMessage(WM_LBUTTONDBLCLK); return CMDIFrameWnd::PreTranslateMessage(pMsg); } 3.在CMainFrame中添加WM_LBUTTONDOWN和WM_LBUTTONDBLCLK消息的响应: void CMainFrame::OnLButtonDown(UINT nFlags, CPoint point) { AfxMessageBox("客户区单击鼠标了!"); CMDIFrameWnd::OnLButtonDblClk(nFlags, point); } void CMainFrame::OnLButtonDblClk(UINT nFlags, CPoint point) { AfxMessageBox("客户区双击鼠标了!"); CMDIFrameWnd::OnLButtonDblClk(nFlags, point); } 4.编译,通过了! >>>>>>>执行,关掉新建的文档,测试结果: #单击鼠标左键:没反应 #双击鼠标左键:没反应 5.将CMainFrame::PreTranslateMessage(MSG* pMsg)中取消对WM_LBUTTONDOWN的注释,编译通过并执行,关掉新建的文档,测试结果: #单击鼠标左键:弹出对话框 #双击鼠标左键:没反应 我不知道该怎么办了,请大家帮助我!!!!!SOS!!! 谢谢大家里!!! huyoo@2004/5/15 --------------------------------------------------- 回复人: enoloo(行者无疆) ( ) 信誉:102 2004-5-15 8:15:59 得分:0 if(pMsg->hwnd==GetActiveFrame() ->GetActiveView()->m_hWnd && pMsg->message==WM_LBUTTONDBLCLK) 正常childframe是不接收dbclick的,他的视图接收。你可以处理视图的dbclick,然后发消息给app,通知建立新文档。 //我搜了,找不到,楼主可否给个连接,谢谢~ --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-5-15 10:23:04 得分:0 测试中... childframe的视图View是接收DBClick,但是我要的不是这个,因为一个childframe就是一个新文档,而不是MDI窗口的客户区. 哇,像 enoloo(行者无疆) 所说,在第二步中那样改,结果在进行第四步:4.编译, >>>>>>>执行,关掉新建的文档,测试结果: 一点ChildFrame的关闭按钮,就出现MDIClientMSG.exe发生错误需要关闭,问我要不要发送报告.$%^$^&*(省去若干抱怨字数...) 我要的效果就是无论在MainFrame中有没有ChildFrame(子文档窗口),只要露出一部分MDIClient客户区,双击那灰色的MDIClient客户区,就弹出打开文件对话框.^_~ //我搜遍了CodeProject和CodeGURU,没有发现有这方面的文章和例子.(!_!) //中国的网站就更不用说了.@_@ 你说它(MDIClient客户区)为什么就能够响应鼠标单击(无论左右键)呢,就是不肯响应左键单击(也是无论左右键)呢? 期待那位大侠解决ing... huyoo@2004/5/15/10:10:18 --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-5-15 10:46:40 得分:0 //寒!!!! //你问的可能是《深入MDI客户窗口编程》(在CSDN中有)的链接吧? //http://expert.csdn.net/Expert/topic/3043/3043812.xml?temp=.7111627 主 题: VC 编程经验总结 1.1 ( VC Programming Skills 1.1 ) 作 者: sgnaw (李逍遥) 里面有一篇《深入MDI客户窗口编程》和一篇《在MDI主框架窗口中添加位图》 去下载吧 我从这里下载的(国内的几个链接好像都下不了): http://lingrer.spymac.net/collection/VC200402.chm &_^########################################### huyoo@2004/5/15 --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-5-15 10:50:15 得分:0 补充:《深入MDI客户窗口编程》和《在MDI主框架窗口中添加位图》这两篇都在--VC/MFC 界面类中. &_^########################################### huyoo@2004/5/15 --------------------------------------------------- 回复人: A_Qiao() ( ) 信誉:100 2004-05-17 16:22:00 得分:0 我仔细读了一下关于鼠标双击消息的说明,发现MSDN中是这样说的 Only windows that have the CS_DBLCLKS style can receive WM_LBUTTONDBLCLK ^^^^^^^^^^^^^^^^^^ messages, which the system generates whenever the user presses, releases, and again presses the left mouse button within the system's double-click time limit. Double-clicking the left mouse button actually generates four messages: WM_LBUTTONDOWN, WM_LBUTTONUP, WM_LBUTTONDBLCLK, and WM_LBUTTONUP again. 因此我认为造成你的问题的原因应该是MDI的客户区窗口类没有CS_DBLCLKS属性,因此它接收不到双击消息。解决的办法就是用函数GetClassLong()和SetClassLong()把CS_DBLCLKS属性给加上。我已经试过了,证实了我的猜测。 --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-21 14:43:00 得分:0 你能够把你的程序具体实现贴上来看一下吗?我在CMainFrame::OnCreate(...)中添加了 DWORD clientstyle; clientstyle=GetClassLong(m_hWndMDIClient,GWL_STYLE); clientstyle|=CS_DBLCLKS; clientstyle=SetClassLong(m_hWndMDIClient,GWL_STYLE,clientstyle); 没有成功 后来改到BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)中 也没有成功!!! //================================================================== 后来我在OnCreateClient中添加下面的: BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext) { lpcs->style|=CS_DBLCLKS; return CMDIFrameWnd::OnCreateClient(lpcs, pContext); } 还会是没有成功,没有响应~~~~~ 请帮帮忙!!! huyoo@2001/05/21 --------------------------------------------------- 回复人: A_Qiao() ( ) 信誉:100 2004-05-22 12:56:00 得分:0 我的实现方法跟你的有所不同,是用SubclassWindow做的,但是原理是一样的. 我仔细看了一下你的方法,觉得问题可能在以下几个方面; (1)在OnCreate的时候,可能m_hWndMDIClient还没有准备好,因此没有成功 (2)如果OnCreate的时候m_hWndMDIClient还没有准备好, 在PreCreateWindow的时候可能也没准备好. (3)在OnCreateClient里,参数LPCREATESTRUCT lpcs中的style是窗口的style, 不要跟窗口类的style搞混了.窗口的style都是WS_开头的,而窗口类的style是CS_开头的. 结论: 应该在OnCreateClient函数中用SetClassLong(), 而不是用lpcs->style|=CS_DBLCLKS --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-23 05:07:00 得分:0 按照A_Qiao()所说,我把CMainFrame::OnCreateClient中的代码改为如下: BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext) { //lpcs->style|=WS_CAPTION;//CS_DBLCLKS; //lpcs->lpszClass=_T("mdiclient"); DWORD clientstyle; clientstyle=::GetClassLong(m_hWndMDIClient,GWL_STYLE); clientstyle|=CS_DBLCLKS; ::SetClassLong(m_hWndMDIClient,GWL_STYLE,clientstyle); return CMDIFrameWnd::OnCreateClient(lpcs, pContext); } 调试通过,但是还是没有反应!!!!!!!! 如果真的象上面的回答那样的话,应该成功的.不过,嘿嘿... 我自己跟踪调试了一下,发现 CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)调用了CMDIFrameWnd::OnCreate(lpCreateStruct), 在CMDIFrameWnd::OnCreate(lpCreateStruct)中又完成了PreCreateWindow和OnCreateClient,所以,说(1)和(2)是不对的. (3)也不能够实现. 因为既然OnCreateClient发生在CMainFrame::OnCreate之前,在OnCreate之后,m_hWndMDIClient还没有准备好, 那么在OnCreateClient中,m_hWndMDIClient还没有准备好就更没有准备好了~~~~~~~~~~~ ######### huyoo@2001/05/23 --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-23 05:21:00 得分:0 补充几点: ## 我在CMainFrame::OnCreateClient中,利用GetClassLong和SetClassLong,想修改MDI客户区的类结构信息,使用了GWL_STYLE参数,犯了一个错误,应该是GCL_STYLE. ##在CMainFrame::OnCreateClient中,我好像没有对函数的参数进行什么修改,可能是不反应的一个原因. ## 在CMainFrame::OnCreateClient中有一段: //lpcs->style|=WS_CAPTION;//CS_DBLCLKS; //lpcs->lpszClass=_T("mdiclient"); 原来我有这么一句:lpcs->style|=WS_VSCROLL,调试的时候,发现在一个内部函数中,主窗口函数不允许MDI 客户区有WS_VSCROLL和WS_HSCROLL(能够上下,左右滚动窗口)的窗口风格,在这里给大家一点提示^_^ 不过我用下面这一句的时候也没有任何变化或者效果产生 lpcs->style|=WS_CAPTION; --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-23 06:46:00 得分:0 ::::::::::::::::::::::::::::对A_Qiao()所说:::::::::::::::: 我又用了子类化去解决: 1.用ClassWizard创建一个基类为CWnd的窗口类,命名为CNewClientWnd. 2.在CMainFrame中添加类型为第1步中创建的窗口类CNewClientWnd的成员变量,命名为m_wndNewClient. 3.在CMainFrame的成员函数OnCreate中,对基类CMDIFrameWnd::OnCreate()的调用之后, 添加一条对SubClassWindow()的调用语句. //========================================== //子类化窗口m_hWndMDIClient if (!m_wndNewClient.SubclassWindow(m_hWndMDIClient)) { TRACE("Failed to subclass MDI client window\n"); return -1;// fail to create } //==================================== m_hWndMDIClient是CMDIFrameWnd的成员变量,包含了MDI客户窗口的句柄。 4.分别添加CNewClientWnd的OnLButtonDown(为了对比用的)和OnLButtonDblClk的消息响应处理: void CNewClientWnd::OnLButtonDblClk(UINT nFlags, CPoint point) { AfxMessageBox("左键在客户区双击了!"); CWnd::OnLButtonDblClk(nFlags, point); } void CNewClientWnd::OnLButtonDown(UINT nFlags, CPoint point) { //AfxMessageBox("左键在客户区按下了!"); CWnd::OnLButtonDown(nFlags, point); } 5.在CMainFrame类中添加CreateClient处理,目的是是客户区窗口能够接受双击: BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext) { DWORD st; st=::GetClassLong(m_wndNewClient.m_hWnd/*m_hWndMDIClient*/,GCL_STYLE); st|=CS_DBLCLKS; ::SetClassLong(m_wndNewClient.m_hWnd/*m_hWndMDIClient*/,GCL_STYLE,st); return CMDIFrameWnd::OnCreateClient(lpcs, pContext); } 6.编译通过,执行,还是没有反应. 请 啊A_QIAO()告诉我,你是怎么样实现的,并且我想知道,你的程序响应了客户区的鼠标双击事件吗? 谢谢!! huyoo@2004/05/23 --------------------------------------------------- 回复人: Mackz(在相互) ( ) 信誉:99 2004-05-23 10:49:00 得分:0 按照《深入MDI客户窗口编程》和《在MDI主框架窗口中添加位图》这两篇文章的介绍,子类化客户区窗口。 注意,基类不是CWnd。 --------------------------------------------------- 回复人: kongyunzhongque(云雀) ( ) 信誉:120 2004-05-23 16:48:00 得分:0 BOOL CNewClientWnd::OnEraseBkgnd(CDC* pDC) { ::SetClassLong(GetSafeHwnd(),GCL_STYLE,::GetClassLong(m_hWnd,GCL_STYLE) | CS_DBLCLKS); 通过验证,可能放的不是地方,不过就是这种思路了 --------------------------------------------------- 回复人: _foo(void) //莫名函数:) ( ) 信誉:163 2004-05-23 17:19:00 得分:0 一两年以前回答过相关问题,好像还进faq了,刚才找了一下找不到 http://www.codeproject.com/dialog/mdibackgroundimage.asp 基本上可以参考这个文章的. 给mdiclient窗口加上CS_DBLCLKS风格并处理消息就可以了. 关键是mdiclient窗口句柄获得的问题 我试过该文代码 HWND hMain = pMainFrame->GetWindow(GW_CHILD)->GetSafeHwnd(); pfnOldWndProc = (WNDPROC)GetWindowLong(hMain, GWL_WNDPROC); SetWindowLong(hMain, GWL_WNDPROC, (long)pfnNewWndProc); 是没问题的, 至于CMDIFrameWnd::m_hWndMDIClient这个成员变量没用过. --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-23 21:43:00 得分:0 //======================= 感谢大家的回答!!! 我测试了你们说的两种方案,发现: ## kongyunzhongque(云雀) 的方案实现的很好,响应了!!我好高兴啊!! 对他所说的位置,我试了另外的地方: [注]我下面测试的,都是测试完一个之后,注释掉,然后进行下一个的测试. 在CNewClientWnd类中: 1.void CNewClientWnd::PreSubclassWindow(){}中,可行~~~,响应了!!!!!!!!! 2.int CNewClientWnd::OnCreate(lpCreateStruct){}中,实现不了,不响应 3.BOOL CNewClientWnd::PreCreateWindow(CREATESTRUCT& cs){}中,实现不了,不响应 4.void CNewClientWnd::OnPaint() { CPaintDC dc(this); // device context for painting ::SetClassLong(GetSafeHwnd(),GCL_STYLE, ::GetClassLong(m_hWnd,GCL_STYLE) | CS_DBLCLKS); // Do not call CWnd::OnPaint() for painting messages } 也实现了,响应!!!~~ 5.void CNewClientWnd::OnShowWindow(bShow,nStatus){}中,实现不了,不响应 其他的,就没有多测试.请大家自己去试试... 最好的位置是在哪里呢,我个人以为既然实现了, 在void CNewClientWnd::PreSubclassWindow(){}中应该是最好了. 在在CMainFrame类中: 我把::SetClassLong(GetSafeHwnd(),GCL_STYLE, ::GetClassLong(m_hWnd,GCL_STYLE) | CS_DBLCLKS); 改为 ::SetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE, ::GetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE) | CS_DBLCLKS); 然后加入到下面的过程中去. 1.BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs){}中,实现不了,不响应 2.int CMainFrame::OnCreate(lpCreateStruct){}中,跟在子类化窗口m_hWndMDIClient语句之后,实现了,响应双击!!!!!!! int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CMDIFrameWnd::OnCreate(lpCreateStruct) == -1) return -1; //========================================== //子类化窗口m_hWndMDIClient if (!m_wndNewClient.SubclassWindow(m_hWndMDIClient)) { TRACE("Failed to subclass MDI client window\n"); return -1;// fail to create } //==================================== ::SetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE, ::GetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE) | CS_DBLCLKS); //==================================== ... } 3.BOOL CMainFrame::OnCreateClient(lpcs, pContext){}中没有实现,不响应. 4.void CMainFrame::OnPaint(){}中,实现了,响应双击!!!!!!! 5.BOOL CMainFrame::OnEraseBkgnd(CDC* pDC){}中,也实现了,响应双击!!!!!!! //=====================================================// ## _foo(void) //莫名函数的方案: [注]我用在那个(不使用子类化的)程序中. 使用下面的代码: HWND hMDIClientArea = GetWindow(GW_CHILD)->GetSafeHwnd(); ::SetClassLong(hMDIClientArea,GCL_STYLE, ::GetClassLong(hMDIClientArea,GCL_STYLE) | CS_DBLCLKS); 然后加入到下面的过程中去. 1.BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs){}中,实现不了,不响应 2.int CMainFrame::OnCreate(lpCreateStruct){}中 int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CMDIFrameWnd::OnCreate(lpCreateStruct) == -1) return -1; HWND hMDIClientArea = GetWindow(GW_CHILD)->GetSafeHwnd(); ::SetClassLong(hMDIClientArea,GCL_STYLE, ::GetClassLong(hMDIClientArea,GCL_STYLE) | CS_DBLCLKS); ... } 实现了,响应双击!!!!!!! 3.BOOL CMainFrame::OnCreateClient(lpcs, pContext){}中没有实现,不响应. 4.void CMainFrame::OnPaint(){}中,实现了,响应双击!!!!!!! 5.BOOL CMainFrame::OnEraseBkgnd(CDC* pDC){}中,也实现了,响应双击!!!!!!! //===================================================// 所以结论是: 无论是子类化的CNewClientWnd,还是CMainFrame,在OnPaint()和OnEraseBkgnd()中,只要对m_hWndMDIClient进行改变类信息操作,都可以实现响应鼠标双击. 在各自的Create事件中,只有在CMainFrame的OnCreate()中修改m_hWndMDIClient的类信息,才可以响应鼠标双击. 到此,我的问题算是得到了解决.还有一些东西需要自己去总结,摸索,和大家交流! 我想我前面没有实现,很有可能是没有取到真正的客户区句柄(m_hWndMDIClient),像GetSafeHwnd()函数,我用的还是不多,今后应该注意;另外一个很大的错误就是把GCL_STYLE(取得类的风格)错成了GWL_STYLE(取出窗口的风格),很长时间得不到结果,值得注意. 另外,在子类化的例子中,下面的代码: DWORD clientstyle; clientstyle=::GetClassLong(m_hWndMDIClient,GCL_STYLE); clientstyle|=CS_DBLCLKS; clientstyle=::SetClassLong(m_hWndMDIClient,GCL_STYLE,clientstyle); 用在CMainFrame的OnCreate()中,和 ::SetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE, ::GetClassLong(m_wndNewClient.m_hWnd,GCL_STYLE) | CS_DBLCLKS); 的效果是一样的~~~~~~~~~~~ 祝大家学习进步,生活快乐,工作顺利!!!! ++++++++++++++++++++++++++++++ huyoo@2004/05/23 ++++++++++++++++++++++++++++++ --------------------------------------------------- 回复人: huyoo(空格键) ( ) 信誉:100 2004-05-23 21:47:00 得分:0 ============================================= =::::::给所有回答我这个问题的专家们:::::::::= ============================================= 首先,感谢你们对我这个问题热情洋溢的回答!解开了让我久久困扰的疑问. 然后,我想说,这不是仅仅对我的问题的解答,而且对将来想实现在客户区响应鼠标双击事件的VC学习者和开发者 也有极大的帮助,你们功不可没!!让他们少走很多弯路. 最后,表示抱歉:我本来以为这是一个很简单的问题,所以只给了20分.没有想到这个问题好久竟然得不到解决,解决的时候,引来了这么多的高手,实在是抱歉,不能给太多的分了!! 今天,我想结束这个帖子了,希望有好办法的人快来发表意见!! 谢谢大家!! ++++++++++++++++++++++++++++++ huyoo@2004/05/23 ++++++++++++++++++++++++++++++ --------------------------------------------------- Copyright ? CSDN.net, Inc. All rights reserved

|