如何捕获VCL没有处理的Windows消息
2007-02-22 01:54
459 查看
---- C++ Builer的VCL提供了对大多数Windows消息的处理机制,这对于一般应用程序是足够了,但VCL也不是无所不包的,对于那些VCL没有处理的Windows消息,在需要时如何进行捕获呢?C++ Builder采用了消息映像表机制,通过消息映像表将特定的Windows消息与代码中的函数联系起来,当窗口捕获到消息时就会调用这个函数,这其实和事件句柄非常相似。
---- C++ Builder消息映像表定义形式如下:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
……
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
END_MESSAGE_MAP(ClassName)
---- ClassName是基础类名,message参数是要捕获的Windows消息,message structure参数用于传递VCL剖析消息后用于传递参数的消息剖析结构名,里面包含处理消息时所需要的全部参数,不同的消息其消息剖析结构是不同的,一般性的消息剖析结构定义如下:
struct Tmessage
{
unsigned int Msg;//Windows消息
long Wparam;
long Lparam;
long Result;
// 本文转自 C++Builder 研究 - http://www.ccrun.com/article.asp?i=226&d=8vzf36 }
---- 这里关键是Result数据成员,它用于设置处理消息后的返回值,程序可以根据不同的返回值进行相应的处理。message handler参数是对消息进行处理的函数名称。
---- 俗话说:千言不如一图。下面就用一个例子具体说明一下,让我们来拖动一个无标题的窗口,实现的原理是通过捕获WM_NCHITTEST消息,在鼠标单击窗体的客户区时发送消息,让Windows认为是个窗体的标题栏上。
---- 创建一个新工程,将单元文件保存为mainform.cpp,工程文件保存为nocaption.bpr。首先得创建一个无标题栏的窗体,将Form1的BorderIcons集合设为空,这就删除了窗口的系统菜单、最大化和最小化按钮,将BorderStyle属性设为bsDialog。注意在设计时创建的所有C++ Builder窗体都会有一个标题栏,你并不能象VB中那样简单地通过将Form1的Caption属性设为空来得到一个无标题栏窗体,你必须在TForm的派生类中重载CreateParams函数,去除相应的窗口标识位,在运行时删去窗体的标题栏。打开mainform.h文件,在private段增加CreateParams的原型:
void __fastcall CreateParams(TCreateParams& Params);
TCreateParams结构包含了创建窗体的各种参数,
在这里我们要屏蔽掉WS_CAPTION位。
在mainform.cpp中增加如下代码:
void __fastcall TForm1::CreateParams(TCreateParams& Params)
{
TForm::CreateParams(Params);//先调用基础类中的成员函数
Params.Style &=~WS_CAPTION;
}
---- 下面创建消息映像表,对WM_NCHITTEST消息进行捕获,在mainform.h的public段输入代码:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER
(WM_NCHITTEST,Tmessage,WMNchitTest)
END_MESSAGE_MAP(TForm)
---- WM_NCHITTEST消息名称为"非客户区命中测试",它优先于所有其它的客户区和非客户区鼠标消息,Windows应用程序通常把这个消息传送给DefWindowProc,然后Windows用这个消息产生基于鼠标位置的所有其它鼠标消息。
---- 在private段加入消息处理函数的声明:
void __fastcall WMNchitTest(Tmessage &message);
---- 在mainform.cpp中输入此函数的定义:
void __fastcall TForm1::WMNchitTest(Tmessage &message)
{
if(GetAsyncKeyState(VK_LBUTTON)< 0)
---- message.Result=HTCAPTION; //如果鼠标左键按下,就通知Windows鼠标所在区域是标题栏区,Windows就会按要求完成拖动操作。
else
message.Result=HTCLIENT;
}
---- 这里用Windows API函数GetAsyncKeyState来实时检查鼠标按钮的状态,通过检测虚拟键码VK_LBUTTON的高位是否被置位来探测鼠标动作,如果按下鼠标左键,此键码的高位被置1,此时GetAsyncKeyState函数将返回一个负数。
---- 按F9编译并运行程序,单击窗体任一位置就可以象单击在标题栏上一样拖动窗体了,这里没有给出结束程序运行的机制,可以通过增加PopMenu来实现。
---- 以上代码在C++ Builder3、Pwin98环境下编译、运行通过。
---- C++ Builder消息映像表定义形式如下:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
……
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
END_MESSAGE_MAP(ClassName)
---- ClassName是基础类名,message参数是要捕获的Windows消息,message structure参数用于传递VCL剖析消息后用于传递参数的消息剖析结构名,里面包含处理消息时所需要的全部参数,不同的消息其消息剖析结构是不同的,一般性的消息剖析结构定义如下:
struct Tmessage
{
unsigned int Msg;//Windows消息
long Wparam;
long Lparam;
long Result;
// 本文转自 C++Builder 研究 - http://www.ccrun.com/article.asp?i=226&d=8vzf36 }
---- 这里关键是Result数据成员,它用于设置处理消息后的返回值,程序可以根据不同的返回值进行相应的处理。message handler参数是对消息进行处理的函数名称。
---- 俗话说:千言不如一图。下面就用一个例子具体说明一下,让我们来拖动一个无标题的窗口,实现的原理是通过捕获WM_NCHITTEST消息,在鼠标单击窗体的客户区时发送消息,让Windows认为是个窗体的标题栏上。
---- 创建一个新工程,将单元文件保存为mainform.cpp,工程文件保存为nocaption.bpr。首先得创建一个无标题栏的窗体,将Form1的BorderIcons集合设为空,这就删除了窗口的系统菜单、最大化和最小化按钮,将BorderStyle属性设为bsDialog。注意在设计时创建的所有C++ Builder窗体都会有一个标题栏,你并不能象VB中那样简单地通过将Form1的Caption属性设为空来得到一个无标题栏窗体,你必须在TForm的派生类中重载CreateParams函数,去除相应的窗口标识位,在运行时删去窗体的标题栏。打开mainform.h文件,在private段增加CreateParams的原型:
void __fastcall CreateParams(TCreateParams& Params);
TCreateParams结构包含了创建窗体的各种参数,
在这里我们要屏蔽掉WS_CAPTION位。
在mainform.cpp中增加如下代码:
void __fastcall TForm1::CreateParams(TCreateParams& Params)
{
TForm::CreateParams(Params);//先调用基础类中的成员函数
Params.Style &=~WS_CAPTION;
}
---- 下面创建消息映像表,对WM_NCHITTEST消息进行捕获,在mainform.h的public段输入代码:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER
(WM_NCHITTEST,Tmessage,WMNchitTest)
END_MESSAGE_MAP(TForm)
---- WM_NCHITTEST消息名称为"非客户区命中测试",它优先于所有其它的客户区和非客户区鼠标消息,Windows应用程序通常把这个消息传送给DefWindowProc,然后Windows用这个消息产生基于鼠标位置的所有其它鼠标消息。
---- 在private段加入消息处理函数的声明:
void __fastcall WMNchitTest(Tmessage &message);
---- 在mainform.cpp中输入此函数的定义:
void __fastcall TForm1::WMNchitTest(Tmessage &message)
{
if(GetAsyncKeyState(VK_LBUTTON)< 0)
---- message.Result=HTCAPTION; //如果鼠标左键按下,就通知Windows鼠标所在区域是标题栏区,Windows就会按要求完成拖动操作。
else
message.Result=HTCLIENT;
}
---- 这里用Windows API函数GetAsyncKeyState来实时检查鼠标按钮的状态,通过检测虚拟键码VK_LBUTTON的高位是否被置位来探测鼠标动作,如果按下鼠标左键,此键码的高位被置1,此时GetAsyncKeyState函数将返回一个负数。
---- 按F9编译并运行程序,单击窗体任一位置就可以象单击在标题栏上一样拖动窗体了,这里没有给出结束程序运行的机制,可以通过增加PopMenu来实现。
---- 以上代码在C++ Builder3、Pwin98环境下编译、运行通过。
相关文章推荐
- 如何捕获VCL没有处理的Windows消息
- 关于捕获VCL没有处理的Windows消息
- 控制台如何像windows程序那样处理消息事件
- QT源码解析(八)Qt是如何处理windows消息的
- 如何在Qt中处理(接收/发送)MFC或Windows消息(直接覆盖MainDialog::nativeEvent,或者QApplication::installNativeEventFilter安装过滤器,或者直接改写QApplication::nativeEventFilter)
- 如何在Qt中处理(接收/发送)MFC或Windows消息(message)
- VCL框架学习之:如何将Windows消息与控件事件挂接起来
- Windows 消息处理机制与事件驱动
- windows消息处理
- MFC如何添加系统消息处理函数?
- Windows的自带控件(比如TButton)大多数消息都由它自己处理,Delphi覆盖了那么多WM_函数优先级较低,一般用不上
- 如何处理自定义消息 (非命令消息)
- Storm 是如何跟踪一条消息以及它衍生出来的消息都被成功处理的
- c#如何处理自定义消息
- 没有外部工具,如何快速发现Windows中毒
- 如何解决Windows1 4000 0下安装Ubuntu16.04双系统后没有Ubuntu引导
- 如何判断所捕获的异常类型,并根据其进行优雅处理
- Windows错误处理及错误消息文字显示
- Android中如何处理未捕获的异常
- Windows 远程桌面出现报错"由于没有远程桌面授权服务器可以提供许可证"的处理