c++ builder 鼠标移开窗体时的事件

如题,还有当鼠标放到窗体上时的事件。请指教。... 如题,还有当鼠标放到窗体上时的事件。请指教。 展开
 我来答
百度网友bfb258cae
2007-12-11 · TA获得超过2378个赞
知道小有建树答主
回答量:755
采纳率:0%
帮助的人:544万
展开全部
在处理窗体消息的时候,我想不少人都会碰到鼠标离开窗体的消息(下面称之为MouseLeave)。在C++ Builder里,并没有直接提供处理这种消息的方法,需要我们自己动手来做。通过参考一些资料,我发现在C++ Builder里面处理MouseLeave,不外乎以下四种方法,现写出来供大家参考。如果有什么不对之处,请指正。

(一)、笨拙的Timer

每每提到捕捉鼠标离开窗体的消息的时候,也许有人就会马上想到用Timer来处理。不错,这种方法很简单,也确实有效。只须在Timer的OnTimer事件中判断鼠标所处位置的坐标是否在窗体内就可以了,详细代码如下:

void __fastcall TForm1::Timer1Timer(TObject *Sender)
{
POINT pt;
GetCursorPos(&pt); //得到鼠标的坐标
RECT rect;
GetWindowRect(Handle,&rect); //得到窗体的矩形范围
if(!PtInRect(rect,pt)) //判断鼠标的坐标是否在窗体的矩形范围内

Caption="out";
else
Caption="in";
}

为什么我要说是笨拙的Timer呢?原因有二:其一、OnTimer是优先级别比较低的消息,从严格意义上讲,上面这种做法并不精准。如果系统正在处理一大堆级别比较高的消息,那我们就无法及时获得MouseLeave消息。其二、Timer是比较宝贵的系统资源,用在MouseLeave上面似乎有些浪费了,因为我们还有更好的方法来做同样的事情。

(二)、霸道的SetCapture()

SetCapture()可以让指定的窗体捕获所有鼠标消息,当然也包括MouseLeave了:

void __fastcall TForm1::FormMouseMove(TObject *Sender, TShiftState Shift,
int X, int Y)
{
Caption="in";
SetCapture(Handle);
TPoint pt(X,Y);
TRect rect;
rect=GetClientRect();
if (!PtInRect(rect,pt))
{
ReleaseCapture();
Caption="out";
}
}

不过这种方法太过于霸道了,因为SetCapture()将所有的鼠标消息据为己有。虽然在捕获了MouseLeave以后已经ReleaseCapture了,但是在捕获过程中,你却无法对其他的鼠标消息做出反应。不信?你不妨在窗体在多放一个Button控件,再运行点点看?:)

(三)、受限的TrackMouseEvent()

MSDN上面说,TrackMouseEvent()可以让指定的窗体接受WM_MOUSELEAVE消息。但是在接受消息以后如果还要继续接受WM_MOUSELEAVE消息,必须重新调用TrackMouseEvent():

//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
: TForm(Owner)
{
MouseTrack=false; //TForm1的私有变量,检测鼠标是否已经被Track
}
//---------------------------------------------------------------------------
void __fastcall TForm1::WndProc(TMessage& Message) //重载WndProc
{
if (Message.Msg==WM_MOUSELEAVE) //在这里捕获WM_MOUSELEAVE消息
{
Caption="out";
MouseTrack=false; //鼠标Track已经完成
}
TForm::WndProc(Message);
}
//---------------------------------------------------------------------------
void __fastcall TForm1::FormMouseMove(TObject *Sender, TShiftState Shift,
int X, int Y)
{
Caption="in";
if (!MouseTrack)
{
TRACKMOUSEEVENT tt;
tt.cbSize=sizeof(tt);
tt.dwFlags=TME_LEAVE;
tt.hwndTrack=Handle;
TrackMouseEvent(&tt);
MouseTrack=true; //开始鼠标Track
}
}
//---------------------------------------------------------------------------

这种方法很好用,唯一的缺点是可能不支持Win98(具体支不支持我也没有做过实验,哪位兄弟有Win98的帮我测试一下)。在Win2000的环境下,我推荐用这种方法,:)

(四)、未知的CM_MOUSELEAVE

在CSDN论坛里经常看到有人说可以通过捕获CM_MOUSELEAVE消息来达到同样的效果。不过根据我的测试,CM_MOUSELEAVE在控件上面工作得很好,可以用来捕获鼠标离开控件的消息。但用在窗体上似乎就不灵验了,可能我自己没有做对吧。如果有哪位大虾知道该怎么用,请告知小弟一声,我将感激不尽。

以上代码均在Win2k Professional+bcb6.0环境中编译成功。
stone_dongdong
2007-12-12 · TA获得超过2853个赞
知道大有可为答主
回答量:1636
采纳率:0%
帮助的人:1028万
展开全部
教你个简单的,使用窗口消息处理函数指针来实现吧
修改窗体的消息处理函数,输入this->。。。,应该可以看到的一个形如
windowproc的函数指针。仿照它的样子再定义一个窗口消息处理函数,要保证参数返回值一模一样的。NewWndProc
同时定义一个函数指针,指向上面所说的函数类型。m_pfnOldWndProc;

然后将此函数指针赋值,指向默认的消息处理函数
m_pfnOldWndProc = this->DefWindowProc;
this->DefWindowProc = NewWndProc;
定义NewWndProc(message .....)
{
....if ( message == WM_KILLFOCUS)
....{ // 此处做鼠标移开的处理
....}
....if ( message == WM_SETFOCUS)
....{ // 此处做鼠标移出的消息处理
....}
....return m_pfnOldWndProc(message...);
}

不好意思,好久没玩BCB了,具体的名称都忘记了,不过上面应该是可行的。

另外,我记得好象,在窗口的函数声明里面可以写消息处理函数(默认未列在窗口属性里面的消息事件处理页面里面的)的,你看看BCB的源代码,我记得以前写过的。
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式