




已阅读5页,还剩12页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
【Visual C+】游戏开发笔记十三 游戏输入消息处理(二) 鼠标消息处理上一节我们讲解了键盘消息处理相关的知识。键盘加鼠标作为目前人机交互方式依旧的主流,在讲完键盘消息处理之后接着讲鼠标消息处理,自然是理所当然的。这一节主要介绍各种鼠标消息的处理方式以及一些相关函数的运用方法,然后用一个小实例来巩固本节所学。一,鼠标消息的处理方式大家都知道,目前市场上主流鼠标规格为两个按键加上一个滚轮。那么,我们先列出Windows中这种鼠标设备输入时的消息:WM_LBUTTONDBLCLK双击鼠标左键消息WM_LBUTTONDOWN单击鼠标左键消息WM_LBUTTONUP松开鼠标左键消息WM_MBUTTONDBLCLK双击鼠标中键(滚轮)消息WM_MBUTTONDOWN单击鼠标中键(滚轮)消息WM_MBUTTONUP松开鼠标中键(滚轮)消息WM_RBUTTONDBLCLK双击鼠标右键消息WM_RBUTTONDOWN单击鼠标右键消息WM_RBUTTONUP松开鼠标右键消息WM_MOUSEMOVE鼠标移动消息WM_MOUSEWHEEL鼠标滚轮转动消息处理鼠标消息的方法与处理键盘消息的方法类似,同样是在消息处理函数中加入要处理的鼠标消息类型,当鼠标消息发生时,输入的参数“wParam”与“lParam”则储存了鼠标状态的相关信息。下面我们分别来展开讲解一下“wParam”与“lParam”参数以及滚轮消息。Param参数lParam参数的值可分为高位字节与低位字节两个部分,其中高节部分储存的是鼠标光标所在的X坐标值,低位字节部分存储的则是鼠标光标所在的Y坐标值。我们可以用下面两个函数来取得鼠标的坐标值:WORDLOWORD(lParam参数);/返回鼠标光标所在的X坐标值WORDHIWORD(lParam参数);/返回鼠标光标所在的Y坐标值这两个两个函数所返回的鼠标光标位置的坐标是相对于内部窗口左上点坐标的。wParam参数wParam参数的值记录着鼠标按键及键盘【Ctrl】键与【Shift】键的状态信息,通过下面的这些定义在“WINUSER.H”中的测试标志与“wParam”参数来检查上述按键的按下状态。MK_LBUTTON按下鼠标右键MK_MBUTTON按下鼠标中(滚轮)键MK_RBUTTON按下鼠标右键MK_SHIFT按下【Shift】键MK_CONTROL按下【Ctrl】键【例子1】例如某一鼠标消息发生时,要测试鼠标左键是否也被按下,程序代码如下:cppview plaincopyprint?1. if(wParam&MK_LBUTTON)/这里应该是按位与&,之前我写错了,谢谢a443475601的指出,2. 3. /鼠标左键被按下4. 这是利用wParam参数与测试标志来测试鼠标键是否被按下的方法。当按键被按下时,条件式“wParam&MK_LBUTTON”所传回的结果会为“true”。当然,若消息函数接收到“WM_LBUTTONDOWN”消息,同样也可以知道鼠标键被按下而不必再去额外做这样的测试。【例子2】如果要测试鼠标左键与【Shift】键的按下状态,那么程序代码如下:cppview plaincopyprint?1. If(wParam&MK_LBUTTON)2. 3. If(wParam&MK_SHIFT)4. 5. /单击鼠标左键6. /按下【Shift】键7. 8. else9. 10. /单击鼠标左键11. /未按下【Shift】键12. 13. 14. else15. 16. If(wParam&MK_SHIFT)17. 18. /未单击鼠标左键19. /按下【Shift】键20. 21. else22. 23. /未单击鼠标左键24. /未按下【Shift】键25. 26. 我们通过这个例子可以清楚,如何利用“wParam”参数与测试标志来测试鼠标键及【Shift】键和【Ctrl】键是否被按下的方法。滚轮消息这里我们要特别提一下鼠标滚轮转动消息(WM_MOUSEWHEEL)。当鼠标滚轮转动消息发生时,“lParam”参数中的值同样是记录光标所在的位置的,而“wParam”参数则分为高位字节与低位字节两部分,低位字节部分跟前面一样是储存鼠标键与【Shift】【Ctrl】键的状态信息的,而高位字节部分的值会是“120”或“-120”。“120”表示鼠标滚轮向前转动,而“-120”则表示向后转动。这里“wParam”高位组值与低位组值所在的函数同样是HIWORD()与LOWORD()。HIWORD(wParam);/高位组,值为“120”或“-120”LOWORD(wParam);/低位组,鼠标键及【Shift】和【Ctrl】键的状态信息二,相关函数的讲解对各种鼠标输入消息及鼠标状态信息的获取方法有了基本认识之后,下面我们将介绍一些游戏程序中以鼠标来做输出设备时常用到的函数。1.获取窗口外鼠标消息的函数为了确保程序可以正确地取得鼠标的输入消息,需要在必要的时候以下面的函数来设定窗口,以取得鼠标在窗口外所发出的消息。HWNDSetCapture(HWNDhWnd);/设定获取窗口外的鼠标消息如果调用了上面的SetCapture()函数,并输入要取得鼠标消息的窗口代号,那么便可取得鼠标在窗口外所发出的消息。这种方法也适用于多窗口的程序,与SetCapture()函数相对应的函数为ReleaseCapture()函数,用于释放窗口取得窗口外鼠标消息的函数。BOOLReleaseCapture(VOID);/释放获取窗口外的鼠标消息2.设定鼠标光标位置的函数BOOLSetCursorPos(intX坐标,intY坐标);/设定鼠标光标位置上面这个SetCursorPos()函数中所设定的坐标是相对于屏幕左上角的屏幕坐标而言。实际上,我们经常需要将这个屏幕坐标转换为游戏窗口中的游戏窗口坐标。因此需要用到API中的一个将窗口坐标转换到屏幕坐标的函数,即ClientToScreen()。屏幕坐标和窗口坐标转换的函数BOOLClientToScreen(HWNDhWnd,/屏幕坐标转换为窗口坐标LPPOINTlpPoint屏幕点坐标);同理,我们得到:窗口坐标转换为屏幕坐标的函数:BOOLScreenToClient(LPPOINTlpPoint窗口点坐标)/窗口坐标转换为屏幕坐标3.显示与隐藏鼠标光标的函数IntShowCursor(BOOLtrue或flase);/隐藏及显示鼠标光标其中,true代表显示光标,false代表隐藏光标。4.限制鼠标光标移动区域的函数WindowsAPI中提供的ClipCursor()函数可以用来设置限制鼠标光标的移动区域和解除鼠标光标移动区域的限制。BOOLClipCursor(CONSTRECT移动区域矩形);/限制鼠标光标移动区域BOOLClipCursor(NOOL);/解除限制这里有一个RECT移动区域矩形,我们在MSDN中找出它的声明:cppview plaincopyprint?1. typedefstructtagRECT2. LONGleft;/矩形区域右上点X坐3. LONGtop;/矩形区域右上点Y坐标4. LONGright;/矩形区域左上点X坐标5. LONGbottom;/矩形区域左上点Y坐标6. RECT;5.取得窗口外部区域及内部区域的API函数我们还需知道取得窗口外部区域及内部区域的API函数。BOOLGetWindowRect(HWNDhWND,LPRECT矩形结构);/取得窗口外部区域矩形BOOLGetClientRect(HWNDhWnd,LPRECT矩形结构体);/取得窗口内部区域矩形这里需要注意的是,GetWindowRect()返回的坐标类型是屏幕坐标。 GetClientRect()返回的坐标类型是窗口坐标。由于限制鼠标光标移动区域的ClipCursor()函数中输入的矩形区域必须是屏幕坐标,因此如果取得的是窗口内部区域,那么还必须将窗口坐标转换为屏幕坐标的操作,下面我们以一段程序代码来说明将鼠标光标限制在窗口内部区域移动的过程:cppview plaincopyprint?1. RECTrect;2. POINTlt,rb;3. GetClientRect(hWnd,&rect);/取得窗口内部矩形4. /将矩形左上点坐标存入lt中5. lt.x=rect.left;6. lt.y=rect.top;7. /将矩形右下坐标存入rb中8. rb.x=rect.right;9. rb.y=rect.bottom;10. /将lt和rb的窗口坐标转换为屏幕坐标11. ClientToScreen(hWnd,);12. ClientToScreen(hWnd,&rb);13. /以屏幕坐标重新设定矩形区域14. rect.left=lt.x;15. rect.top=lt.y;16. rect.right=rb.x;17. rect.bottom=rb.y;18. /限制鼠标光标移动区域19. ClipCursor(&rect);三,在实例中融会贯通讲了这么多的windowsAPI函数了,也早该到了我们的showtime了,依然,我们通过一个实例来把本节所讲的内容融会贯通。这个实例处理鼠标移动消息使飞机在窗口中移动,并且处理单击鼠标左键消息来让飞机发射子弹,而且设定了鼠标光标的位置,隐藏了鼠标光标,还有限制了鼠标光标移动的区域。(背景贴图采用循环背景滚动,其实很简单,就是每次都把窗口右边多余的部分再贴到窗口坐标来,以后有机会我会作为一节笔记具体讲)同样的,我们贴出详细注释好的代码:cppview plaincopyprint?1. #includestdafx.h2. #include3. 4. /定义结构体5. structBULLET/bullet结构体代表飞机子弹6. 7. intx,y;/子弹坐标8. boolexist;/子弹是否存在9. ;10. 11. /全局变量声明12. HINSTANCEhInst;13. HBITMAPbg,ship,bullet;/存储背景图,飞机图,子弹图14. HDChdc,mdc,bufdc;15. HWNDhWnd;16. DWORDtPre,tNow;17. intx,y,nowX,nowY;/x,y代表鼠标光标所在位置,nowX,nowY代表飞机坐标,也是贴图的位置18. intw=0,bcount;/w为滚动背景所要裁剪的区域宽度,bcount记录飞机现有子弹数目19. BULLETb30;/声明一个“bullet”类型的数组,用来存储飞机发出的子弹20. 21. /全局函数声明22. ATOMMyRegisterClass(HINSTANCEhInstance);23. BOOLInitInstance(HINSTANCE,int);24. LRESULTCALLBACKWndProc(HWND,UINT,WPARAM,LPARAM);25. voidMyPaint(HDChdc);26. 27. /*WinMain函数,程序入口点函数*28. intAPIENTRYWinMain(HINSTANCEhInstance,29. HINSTANCEhPrevInstance,30. LPSTRlpCmdLine,31. intnCmdShow)32. 33. MSGmsg;34. 35. MyRegisterClass(hInstance);36. 37. /初始化38. if(!InitInstance(hInstance,nCmdShow)39. 40. returnFALSE;41. 42. 43. /消息循环44. GetMessage(&msg,NULL,NULL,NULL);/初始化msg45. while(msg.message!=WM_QUIT)46. 47. if(PeekMessage(&msg,NULL,0,0,PM_REMOVE)48. 49. TranslateMessage(&msg);50. DispatchMessage(&msg);51. 52. else53. 54. tNow=GetTickCount();55. if(tNow-tPre=40)56. MyPaint(hdc);57. 58. 59. 60. returnmsg.wParam;61. 62. 63. /*设计一个窗口类,类似填空题,使用窗口结构体*64. ATOMMyRegisterClass(HINSTANCEhInstance)65. 66. WNDCLASSEXwcex;67. 68. wcex.cbSize=sizeof(WNDCLASSEX);69. wcex.style=CS_HREDRAW|CS_VREDRAW|CS_DBLCLKS;70. wcex.lpfnWndProc=(WNDPROC)WndProc;71. wcex.cbClsExtra=0;72. wcex.cbWndExtra=0;73. wcex.hInstance=hInstance;74. wcex.hIcon=NULL;75. wcex.hCursor=NULL;76. wcex.hCursor=LoadCursor(NULL,IDC_ARROW);77. wcex.hbrBackground=(HBRUSH)(COLOR_WINDOW+1);78. wcex.lpszMenuName=NULL;79. wcex.lpszClassName=canvas;80. wcex.hIconSm=NULL;81. 82. returnRegisterClassEx(&wcex);83. 84. 85. /*初始化函数*86. /1.设定飞机初始位置87. /2.设定鼠标光标位置及隐藏88. /3.限制鼠标光标移动区域89. BOOLInitInstance(HINSTANCEhInstance,intnCmdShow)90. 91. HBITMAPbmp;92. POINTpt,lt,rb;93. RECTrect;94. 95. hInst=hInstance;96. 97. hWnd=CreateWindow(canvas,绘图窗口,WS_OVERLAPPEDWINDOW,98. CW_USEDEFAULT,0,CW_USEDEFAULT,0,NULL,NULL,hInstance,NULL);99. 100. if(!hWnd)101. 102. returnFALSE;103. 104. 105. MoveWindow(hWnd,10,10,640,480,true);106. ShowWindow(hWnd,nCmdShow);107. UpdateWindow(hWnd);108. 109. hdc=GetDC(hWnd);110. mdc=CreateCompatibleDC(hdc);111. bufdc=CreateCompatibleDC(hdc);112. 113. bmp=CreateCompatibleBitmap(hdc,640,480);114. SelectObject(mdc,bmp);115. 116. bg=(HBITMAP)LoadImage(NULL,bg.bmp,IMAGE_BITMAP,648,480,LR_LOADFROMFILE);117. ship=(HBITMAP)LoadImage(NULL,ship.bmp,IMAGE_BITMAP,100,148,LR_LOADFROMFILE);118. bullet=(HBITMAP)LoadImage(NULL,bullet.bmp,IMAGE_BITMAP,10,20,LR_LOADFROMFILE);119. 120. /设定鼠标光标的x,y值,并设定飞机贴图坐标的“nowX”和“nowY”的值为(300,300)121. x=300;122. y=300;123. nowX=300;124. nowY=300;125. 126. /设定光标位置127. pt.x=300;128. pt.y=300;129. ClientToScreen(hWnd,&pt);130. SetCursorPos(pt.x,pt.y);131. 132. ShowCursor(false);/隐藏鼠标光标133. 134. /限制鼠标光标移动区域135. GetClientRect(hWnd,&rect);/取得窗口内部矩形136. /将矩形左上点坐标存入lt中137. lt.x=rect.left;138. lt.y=rect.top;139. /将矩形右下坐标存入rb中140. rb.x=rect.right;141. rb.y=rect.bottom;142. /将lt和rb的窗口坐标转换为屏幕坐标143. ClientToScreen(hWnd,);144. ClientToScreen(hWnd,&rb);145. /以屏幕坐标重新设定矩形区域146. rect.left=lt.x;147. rect.top=lt.y;148. rect.right=rb.x;149. rect.bottom=rb.y;150. /限制鼠标光标移动区域151. ClipCursor(&rect);152. 153. MyPaint(hdc);154. 155. returnTRUE;156. 157. 158. /*自定义绘图函数*159. /1.设定飞机坐标并进行贴图160. /2.设定所有子弹坐标并进行贴图161. /3.显示真正的鼠标光标所在坐标162. voidMyPaint(HDChdc)163. 164. charstr20=;165. inti;166. 167. /贴上背景图168. SelectObject(bufdc,bg);169. BitBlt(mdc,0,0,w,480,bufdc,640-w,0,SRCCOPY);170. BitBlt(mdc,w,0,640-w,480,bufdc,0,0,SRCCOPY);171. 172. /计算飞机的贴图坐标,设定每次进行飞机贴图时,其贴图坐标(nowX,nowY)会以10个单位慢慢向鼠标光标所在的目的点(x,y)接近,直到两个坐标相同为止173. if(nowXx)177. nowX=x;178. 179. else180. 181. nowX-=10;182. if(nowXx)183. nowX=x;184. 185. 186. if(nowYy)190. nowY=y;191. 192. else193. 194. nowY-=10;195. if(nowYy)196. nowY=y;197. 198. 199. /贴上飞机图200. SelectObject(bufdc,ship);201. BitBlt(mdc,nowX,nowY,100,74,bufdc,0,74,SRCAND);202. BitBlt(mdc,nowX,nowY,100,74,bufdc,0,0,SRCPAINT);203. 204. /子弹的贴图,先判断子弹数目“bcount”的值是否为“0”。若不为0,则对子弹数组中各个还存在的子弹按照其所在的坐标(bi.x,bi.y)循环进行贴图操作205. SelectObject(bufdc,bullet);206. if(bcount!=0)207. for(i=0;i30;i+)208. if(bi.exist)209. 210. /贴上子弹图211. BitBlt(mdc,bi.x,bi.y,10,10,bufdc,0,10,SRCAND);212. BitBlt(mdc,bi.x,bi.y,10,10,bufdc,0,0,SRCPAINT);213. 214. /设置下一个子弹的坐标。子弹是又右向左发射的,因此,每次其X轴上的坐标值递减10个单位,这样贴图会产生往左移动的效果。而如果子弹下次的坐标已超出窗口的可见范围(hi.x0),那么子弹设为不存在,并将子弹总数bcount变量值减1.215. bi.x-=10;216. if(bi.x0)217. 218. bcount-;219. bi.exist=false;220. 221. 222. 223. /显示鼠标坐标224. sprintf(str,鼠标X坐标为%d,x);225. TextOut(mdc,0,0,str,strlen(str);226. sprintf(str,鼠标Y坐标为%d,y);227. TextOut(mdc,0,20,str,strlen(str);228. 229. BitBlt(hdc,0,0,640,480,mdc,0,0,SRCCOPY);230. 231. tPre=GetTickCount();232. 233. w+=10;234. if(w=640)235. w=0;236. 237. 238. /*消息处理函数*239. /1.处理WM_LBUTTONDOWN消息发射子弹240. /2.处理WM_MOUSEMOVE消息设定飞机贴图坐标241. /3.在窗口结束时恢复鼠标移动区域242. LRESULTCALLBACKWndProc(HWNDhWnd,UINTmessage,WPARAMwParam,LPARAMlParam)243. 244. inti;245. 246. switch(message)247. 248. caseWM_KEYDOWN:/按键按下消息249. if(wParam=VK_ESCAPE)/按下【Esc】键250. PostQuitMessage(0);251. break
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 乳酸菌素片课件
- 护理导论系统的基本属性
- 2025年初级招采人员考试(招标采购专业实务)在线模拟题库及答案及答案(辽宁省)
- 铁建项目工会汇报
- 公司法自考课件
- 颅骨缺损修补术的护理
- 公司治理李维安课件
- 护理计划的实施与评价
- 劳动合同范文:安全生产责任状8篇
- 2025年设备中介合同范本
- 中班健康魔法消气屋课件
- 腕部损伤的护理课件
- 蓝莓水肥一体化栽培技术规程
- 【基于Creo的NGW型行星齿轮减速器设计9000字】
- 营销沟通技巧培训
- DB64∕T 2023-2024 不动产登记操作指南
- oa数据安全管理制度
- 中医诊所消毒管理制度
- 旋风除尘器设计选型
- 2026届新高考语文热点复习小说阅读
- 2025年鱼菜共生项目可行性研究报告
评论
0/150
提交评论