如何在MDI客户区窗口响应鼠标双击消息?

<script type="text/javascript"> </script> <script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
<script type="text/javascript"> </script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>

    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");  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>
<script type="text/javascript"> </script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值