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

类别:VC语言 点击:0 评论:0 推荐:

 
  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
 

本文地址:http://com.8s8s.com/it/it792.htm