-
c语言入门之用游戏操纵杆摸拟鼠标
要编写一个支持游戏操纵杆的应用程序,首先必须要捕获游戏操纵杆,接着要处理Windows发送给程序窗口的操纵杆消息,最后使用完操纵杆后,还应将捕获的操纵杆资源释放。
调用API函数joySetCapture能捕获游戏操纵杆。调用joySetCapture函数后,操纵杆产生的所有消息将会发送到指定的窗口。它的原型为:
MMRESULT joySetCapture(HWND hwnd, UINT uJoyID, UINT uPeriod, BOOL fChanged );
其中,参数hwnd为接收操纵杆消息的窗口句柄;参数uJoyID为要捕获的操纵杆标识,它可以是JOYSTICKID1或是JOYSTICKID2,即第一、第二个游戏操纵杆;参数uPeriod为轮询的频率,单位为毫秒,它指定给应用程序发送有关操纵杆信息的间隔时间;参数fChanged为改变位置标识,可设为false。
要释放操纵杆的捕获时,使用joyReleaseCapture函数。它只有一个参数,就是操纵杆的标识JOYSTICKID1或JOYSTICKID2。
下面,就让我们用Borland C++ Builder 5.0来做一个用游戏操纵杆模拟鼠标的程序。
运行Borland C++ Builder 5.0,双击窗体Form1,在Form1的OnCreate事件中加入以下代码捕获一个游戏操纵杆:
void __fastcall TForm1::FormCreate(TObject *Sender)
{
int JoyMsg;
//捕获游戏操纵杆
JoyMsg=joySetCapture(Handle,JOYSTICKID1,0,false);
if(JoyMsg==JOYERR_NOCANDO)
{
//捕获失败
ShowMessage("不能捕获游戏杆!");
}
else
{
if(JoyMsg==JOYERR_UNPLUGGED)
{
//没有连接
ShowMessage("游戏杆未与系统连接!");
}
else
{
if(JoyMsg==MMSYSERR_NODRIVER)
{
//没有安装
ShowMessage("系统没有安装游戏杆!");
}
else
{
//捕获成功
ShowMessage("捕获游戏杆成功!");
}}}}
在Form1的OnCloseQuery事件中加入代码,让程序关闭时释放操纵杆捕获的资源:
void __fastcall TForm1::FormCloseQuery(TObject *Sender, bool &CanClose)
{
//释放操纵杆捕获
joyReleaseCapture(JOYSTICKID1);
}
捕获游戏操纵杆后,Windows会把所有的操纵杆消息发送给窗口Form1。当操纵杆的方向钮按被按下时,产生的是MM_JOY1MOVE消息,当功能按钮被按下时,产生MM_JOY1BUTTONDOWN消息。在程序中分别响应并处理这两个消息,就可以模拟鼠标的移动和点击。
但是在C++ Builder中,这两条消息并不是标准的Windows消息,这就需要我们自已定义和处理消息了。在C++ Builder里响应自定义消息的步骤为:
1.建立消息映射表
2.声明消息处理函数
3.编写消息处理函数
首先在代码编辑窗口点击右键,选择弹出菜单的“Open Source/Header File”或是按热键Ctrl+F6,打开窗体Form1头文件“Uint1.h”。
在窗体的TForm1类中的公有成员中加入代码来建立消息映射表,把消息的处理权交给自定义的消息处理函数:
public:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER(MM_JOY1BUTTONDOWN,TMessage,OnJoyDown)
MESSAGE_HANDLER(MM_JOY1MOVE,TMessage,OnJoyMove)
END_MESSAGE_MAP(TForm)
然后在类的私有成员中加入代码声明消息处理函数:
private:
void __fastcall OnJoyDown(TMessage &Message);
void __fastcall OnJoyMove(TMessage &Message);
最后,按Ctrl+F6键切换回“Uint1.cpp”的编辑窗口,在末尾空白处添加下面两个自定义的消息响应函数:
//自定义的MM_JOY1BUTTONDOWN消息响应函数OnJoyDown
void __fastcall TForm1::OnJoyDown(TMessage &Message)
{
if(Message.WParam & JOY_BUTTON1)
{
//模拟鼠标左键按下
mouse_event(MOUSEEVENTF_LEFTDOWN,0,0,0,0);
Caption="左键按下";
}
if(Message.WParam & JOY_BUTTON2)
{
//模拟鼠标右键按下
mouse_event(MOUSEEVENTF_RIGHTDOWN,0,0,0,0);
Caption="右键按下";
}
if(Message.WParam & JOY_BUTTON3)
{
//模拟鼠标左键抬起
mouse_event(MOUSEEVENTF_LEFTUP,0,0,0,0);
Caption="左键抬起";
}
if(Message.WParam & JOY_BUTTON4)
{
//模拟鼠标右键抬起
mouse_event(MOUSEEVENTF_RIGHTUP,0,0,0,0);
Caption="右键抬起";
}
//继续传递消息
TForm::Dispatch(&Message);
}
//自定义的MM_JOY1MOVE消息响应函数OnJoyDown
void __fastcall TForm1::OnJoyMove(TMessage &Message)
{
int x,y;
POINT pt;
//取得鼠标当前坐标
GetCursorPos(&pt);
x=LOWORD(Message.LParam);
y=HIWORD(Message.LParam);
if(x!=32678)
{
if(x)
{
//向右
pt.x+=10;
}
else
{
//向左
pt.x-=10;
}}
if(y!=32678)
{
if(y)
{
//向下
pt.y+=10;
}
else
{
//向上
pt.y-=10;
}}
//设置鼠标坐标
SetCursorPos(pt.x,pt.y);
//继续传递消息
TForm::Dispatch(&Message);
}
注意:调试运行这个程序,系统必须要安装有游戏操纵杆。自定义的消息处理函数末尾最好加一句 TForm1::Dispatch(&Message),这条语句的作用是让消息继续传递下去。Windows是使用用消息处理机制的,如果没有这一句语句,消息将完全被拦截,Windows程序可能由于得不到消息而无法实现正常的功能。
调用API函数joySetCapture能捕获游戏操纵杆。调用joySetCapture函数后,操纵杆产生的所有消息将会发送到指定的窗口。它的原型为:
MMRESULT joySetCapture(HWND hwnd, UINT uJoyID, UINT uPeriod, BOOL fChanged );
其中,参数hwnd为接收操纵杆消息的窗口句柄;参数uJoyID为要捕获的操纵杆标识,它可以是JOYSTICKID1或是JOYSTICKID2,即第一、第二个游戏操纵杆;参数uPeriod为轮询的频率,单位为毫秒,它指定给应用程序发送有关操纵杆信息的间隔时间;参数fChanged为改变位置标识,可设为false。
要释放操纵杆的捕获时,使用joyReleaseCapture函数。它只有一个参数,就是操纵杆的标识JOYSTICKID1或JOYSTICKID2。
下面,就让我们用Borland C++ Builder 5.0来做一个用游戏操纵杆模拟鼠标的程序。
运行Borland C++ Builder 5.0,双击窗体Form1,在Form1的OnCreate事件中加入以下代码捕获一个游戏操纵杆:
void __fastcall TForm1::FormCreate(TObject *Sender)
{
int JoyMsg;
//捕获游戏操纵杆
JoyMsg=joySetCapture(Handle,JOYSTICKID1,0,false);
if(JoyMsg==JOYERR_NOCANDO)
{
//捕获失败
ShowMessage("不能捕获游戏杆!");
}
else
{
if(JoyMsg==JOYERR_UNPLUGGED)
{
//没有连接
ShowMessage("游戏杆未与系统连接!");
}
else
{
if(JoyMsg==MMSYSERR_NODRIVER)
{
//没有安装
ShowMessage("系统没有安装游戏杆!");
}
else
{
//捕获成功
ShowMessage("捕获游戏杆成功!");
}}}}
在Form1的OnCloseQuery事件中加入代码,让程序关闭时释放操纵杆捕获的资源:
void __fastcall TForm1::FormCloseQuery(TObject *Sender, bool &CanClose)
{
//释放操纵杆捕获
joyReleaseCapture(JOYSTICKID1);
}
捕获游戏操纵杆后,Windows会把所有的操纵杆消息发送给窗口Form1。当操纵杆的方向钮按被按下时,产生的是MM_JOY1MOVE消息,当功能按钮被按下时,产生MM_JOY1BUTTONDOWN消息。在程序中分别响应并处理这两个消息,就可以模拟鼠标的移动和点击。
但是在C++ Builder中,这两条消息并不是标准的Windows消息,这就需要我们自已定义和处理消息了。在C++ Builder里响应自定义消息的步骤为:
1.建立消息映射表
2.声明消息处理函数
3.编写消息处理函数
首先在代码编辑窗口点击右键,选择弹出菜单的“Open Source/Header File”或是按热键Ctrl+F6,打开窗体Form1头文件“Uint1.h”。
在窗体的TForm1类中的公有成员中加入代码来建立消息映射表,把消息的处理权交给自定义的消息处理函数:
public:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER(MM_JOY1BUTTONDOWN,TMessage,OnJoyDown)
MESSAGE_HANDLER(MM_JOY1MOVE,TMessage,OnJoyMove)
END_MESSAGE_MAP(TForm)
然后在类的私有成员中加入代码声明消息处理函数:
private:
void __fastcall OnJoyDown(TMessage &Message);
void __fastcall OnJoyMove(TMessage &Message);
最后,按Ctrl+F6键切换回“Uint1.cpp”的编辑窗口,在末尾空白处添加下面两个自定义的消息响应函数:
//自定义的MM_JOY1BUTTONDOWN消息响应函数OnJoyDown
void __fastcall TForm1::OnJoyDown(TMessage &Message)
{
if(Message.WParam & JOY_BUTTON1)
{
//模拟鼠标左键按下
mouse_event(MOUSEEVENTF_LEFTDOWN,0,0,0,0);
Caption="左键按下";
}
if(Message.WParam & JOY_BUTTON2)
{
//模拟鼠标右键按下
mouse_event(MOUSEEVENTF_RIGHTDOWN,0,0,0,0);
Caption="右键按下";
}
if(Message.WParam & JOY_BUTTON3)
{
//模拟鼠标左键抬起
mouse_event(MOUSEEVENTF_LEFTUP,0,0,0,0);
Caption="左键抬起";
}
if(Message.WParam & JOY_BUTTON4)
{
//模拟鼠标右键抬起
mouse_event(MOUSEEVENTF_RIGHTUP,0,0,0,0);
Caption="右键抬起";
}
//继续传递消息
TForm::Dispatch(&Message);
}
//自定义的MM_JOY1MOVE消息响应函数OnJoyDown
void __fastcall TForm1::OnJoyMove(TMessage &Message)
{
int x,y;
POINT pt;
//取得鼠标当前坐标
GetCursorPos(&pt);
x=LOWORD(Message.LParam);
y=HIWORD(Message.LParam);
if(x!=32678)
{
if(x)
{
//向右
pt.x+=10;
}
else
{
//向左
pt.x-=10;
}}
if(y!=32678)
{
if(y)
{
//向下
pt.y+=10;
}
else
{
//向上
pt.y-=10;
}}
//设置鼠标坐标
SetCursorPos(pt.x,pt.y);
//继续传递消息
TForm::Dispatch(&Message);
}
注意:调试运行这个程序,系统必须要安装有游戏操纵杆。自定义的消息处理函数末尾最好加一句 TForm1::Dispatch(&Message),这条语句的作用是让消息继续传递下去。Windows是使用用消息处理机制的,如果没有这一句语句,消息将完全被拦截,Windows程序可能由于得不到消息而无法实现正常的功能。
最新更新
Objective-C语法之代码块(block)的使用
VB.NET eBook
Add-in and Automation Development In VB.NET 2003 (F
Add-in and Automation Development In VB.NET 2003 (8
Add-in and Automation Development in VB.NET 2003 (6
Add-in and Automation Development In VB.NET 2003 (5
AddIn Automation Development In VB.NET 2003 (4)
AddIn And Automation Development In VB.NET 2003 (2)
Addin and Automation Development In VB.NET 2003 (3)
AddIn And Automation Development In VB.NET 2003 (1)
2个场景实例讲解GaussDB(DWS)基表统计信息估
常用的 SQL Server 关键字及其含义
动手分析SQL Server中的事务中使用的锁
openGauss内核分析:SQL by pass & 经典执行
一招教你如何高效批量导入与更新数据
天天写SQL,这些神奇的特性你知道吗?
openGauss内核分析:执行计划生成
[IM002]Navicat ODBC驱动器管理器 未发现数据
初入Sql Server 之 存储过程的简单使用
SQL Server -- 解决存储过程传入参数作为s
武装你的WEBAPI-OData入门
武装你的WEBAPI-OData便捷查询
武装你的WEBAPI-OData分页查询
武装你的WEBAPI-OData资源更新Delta
5. 武装你的WEBAPI-OData使用Endpoint 05-09
武装你的WEBAPI-OData之API版本管理
武装你的WEBAPI-OData常见问题
武装你的WEBAPI-OData聚合查询
OData WebAPI实践-OData与EDM
OData WebAPI实践-Non-EDM模式