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(空格键) ( ) 信誉: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(空格键) ( ) 信誉:100 2004-5-15 10:50:15 得分:0 补充:《深入MDI客户窗口编程》和《在MDI主框架窗口中添加位图》这两篇都在--VC/MFC 界面类中.&_^###########################################
--------------------------------------------------- 回复人: 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);}
还会是没有成功,没有响应~~~~~
请帮帮忙!!!
--------------------------------------------------- 回复人: 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"); 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了,刚才找了一下找不到MDIbackgroundimage.asp">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"); 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
<script type="text/javascript"> </script> <script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>