微软鼠标滑轮不能用1000 QQ聊天面板滑轮用不了如何解决?

21573人阅读
【Visual C++】(39)
【Visual C++】游戏开发(56)
本系列文章由zhmxy555编写,转载请注明出处。&作者:毛星云&&&&邮箱:&&&&&欢迎邮件交流编程心得上一节我们讲解了键盘消息处理相关的知识。键盘加鼠标作为目前人机交互方式依旧的主流,在讲完键盘消息处理之后接着讲鼠标消息处理,自然是理所当然的。&这一节主要介绍各种鼠标消息的处理方式以及一些相关函数的运用方法,然后用一个小实例来巩固本节所学。一,鼠标消息的处理方式大家都知道,目前市场上主流鼠标规格为两个按键加上一个滚轮。那么,我们先列出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”参数以及滚轮消息。&l&Param参数lParam参数的值可分为高位字节与低位字节两个部分,其中高节部分储存的是鼠标光标所在的X坐标值,低位字节部分存储的则是鼠标光标所在的Y坐标值。我们可以用下面两个函数来取得鼠标的坐标值:WORD&LOWORD(lParam参数);&&&&&&&&&&&//返回鼠标光标所在的X坐标值WORD&HIWORD(lParam参数);&&&&&&&&&&&//返回鼠标光标所在的Y坐标值这两个两个函数所返回的鼠标光标位置的坐标是相对于内部窗口左上点坐标的。&2&wParam参数&&wParam&参数的值记录着鼠标按键及键盘【Ctrl】键与【Shift】键的状态信息,通过下面的这些定义在“WINUSER.H”中的测试标志与“wParam”参数来检查上述按键的按下状态。MK_LBUTTON&&&&&&&&&&&&按下鼠标右键MK_MBUTTON&&&&&&&&&&&按下鼠标中(滚轮)键MK_RBUTTON&&&&&&&&&&&&按下鼠标右键MK_SHIFT&&&&&&&&&&&&&&&&按下【Shift】键MK_CONTROL&&&&&&&&&&&&按下【Ctrl】键【例子1】例如某一鼠标消息发生时,要测试鼠标左键是否也被按下,程序代码如下:if(wParam & MK_LBUTTON)
//这里应该是按位与&,之前我写错了,谢谢 a 的指出,
//鼠标左键被按下
}这是利用wParam参数与测试标志来测试鼠标键是否被按下的方法。当按键被按下时,条件式“wParam&&&&MK_LBUTTON”所传回的结果会为“true”。当然,若消息函数接收到“WM_LBUTTONDOWN”消息,同样也可以知道鼠标键被按下而不必再去额外做这样的测试。【例子2】如果要测试鼠标左键与【Shift】键的按下状态,那么程序代码如下:If(wParam & MK_LBUTTON)
If(wParam & MK_SHIFT)
//单击鼠标左键
//按下【Shift】键
//单击鼠标左键
//未按下【Shift】键
If(wParam & MK_SHIFT)
//未单击鼠标左键
//按下【Shift】键
//未单击鼠标左键
//未按下【Shift】键
}我们通过这个例子可以清楚,如何利用“wParam”参数与测试标志来测试鼠标键及【Shift】键和【Ctrl】键是否被按下的方法。&3&滚轮消息这里我们要特别提一下鼠标滚轮转动消息(WM_MOUSEWHEEL)。当鼠标滚轮转动消息发生时,“lParam”参数中的值同样是记录光标所在的位置的,而“wParam”参数则分为高位字节与低位字节两部分,低位字节部分跟前面一样是储存鼠标键与【Shift】【Ctrl】键的状态信息的,而高位字节部分的值会是“120”或“-120”。“120”表示鼠标滚轮向前转动,而“-120”则表示向后转动。这里“wParam”高位组值与低位组值所在的函数同样是HIWORD(&)与LOWORD(&)。HIWORD(wParam);//高位组,值为“120”或“-120”LOWORD(wParam);//低位组,鼠标键及【Shift】和【Ctrl】键的状态信息二,相关函数的讲解对各种鼠标输入消息及鼠标状态信息的获取方法有了基本认识之后,下面我们将介绍一些游戏程序中以鼠标来做输出设备时常用到的函数。1.获取窗口外鼠标消息的函数为了确保程序可以正确地取得鼠标的输入消息,需要在必要的时候以下面的函数来设定窗口,以取得鼠标在窗口外所发出的消息。HWND&SetCapture(HWND&hWnd)&;&&&//设定获取窗口外的鼠标消息如果调用了上面的SetCapture(&)函数,并输入要取得鼠标消息的窗口代号,那么便可取得鼠标在窗口外所发出的消息。这种方法也适用于多窗口的程序,与SetCapture(&)函数相对应的函数为ReleaseCapture(&)函数,用于释放窗口取得窗口外鼠标消息的函数。BOOL&ReleaseCapture(VOID);&&&&//释放获取窗口外的鼠标消息2.设定鼠标光标位置的函数BOOL&SetCursorPos(int&X坐标,int&Y坐标);&&&&&//设定鼠标光标位置上面这个SetCursorPos()函数中所设定的坐标是相对于屏幕左上角的屏幕坐标而言。实际上,我们经常需要将这个屏幕坐标转换为游戏窗口中的游戏窗口坐标。因此需要用到API中的一个将窗口坐标转换到屏幕坐标的函数,即ClientToScreen()。屏幕坐标和窗口坐标转换的函数BOOL&ClientToScreen(HWND&hWnd,&&&&&//屏幕坐标转换为窗口坐标LPPOINT&lpPoint屏幕点坐标);&&&同理,我们得到:窗口坐标转换为屏幕坐标的函数:BOOL&ScreenToClient(&&&LPPOINT&lpPoint窗口点坐标&)&&&//窗口坐标转换为屏幕坐标3.显示与隐藏鼠标光标的函数Int&ShowCursor(BOOL&true或flase);&&&&//隐藏及显示鼠标光标其中,true代表显示光标,false代表隐藏光标。4.限制鼠标光标移动区域的函数Windows&API中提供的ClipCursor()函数可以用来设置限制鼠标光标的移动区域和解除鼠标光标移动区域的限制。BOOL&ClipCursor(CONST&RECT&移动区域矩形);&&//限制鼠标光标移动区域BOOL&ClipCursor(NOOL);&&&&&&&&&&&&&&&&&&&&&//解除限制这里有一个RECT移动区域矩形,我们在MSDN中找出它的声明:typedef struct tagRECT {
//矩形区域右上点X坐
//矩形区域右上点Y坐标
//矩形区域左上点X坐标
//矩形区域左上点Y坐标
} RECT;5.取得窗口外部区域及内部区域的API函数我们还需知道取得窗口外部区域及内部区域的API函数。BOOL&GetWindowRect(HWND&hWND,LPRECT&矩形结构);//取得窗口外部区域矩形BOOL&GetClientRect(HWND&hWnd,LPRECT&矩形结构体);&//取得窗口内部区域矩形这里需要注意的是,GetWindowRect()返回的坐标类型是屏幕坐标。&&
& & &GetClientRect()返回的坐标类型是窗口坐标。由于限制鼠标光标移动区域的ClipCursor()函数中输入的矩形区域必须是屏幕坐标,因此如果取得的是窗口内部区域,那么还必须将窗口坐标转换为屏幕坐标的操作,下面我们以一段程序代码来说明将鼠标光标限制在窗口内部区域移动的过程:RECT rect;
POINT lt,rb;
GetClientRect(hWnd,&rect);
//取得窗口内部矩形
//将矩形左上点坐标存入lt中
lt.x = rect.
lt.y = rect.
//将矩形右下坐标存入rb中
rb.x = rect.
rb.y = rect.
//将lt和rb的窗口坐标转换为屏幕坐标
ClientToScreen(hWnd,&);
ClientToScreen(hWnd,&rb);
//以屏幕坐标重新设定矩形区域
rect.left = lt.x;
rect.top = lt.y;
rect.right = rb.x;
rect.bottom = rb.y;
//限制鼠标光标移动区域
ClipCursor(&rect);三,在实例中融会贯通讲了这么多的windows&API函数了,也早该到了我们的showtime了,依然,我们通过一个实例来把本节所讲的内容融会贯通。这个实例处理鼠标移动消息使飞机在窗口中移动,并且处理单击鼠标左键消息来让飞机发射子弹,而且设定了鼠标光标的位置,隐藏了鼠标光标,还有限制了鼠标光标移动的区域。(背景贴图采用循环背景滚动,其实很简单,就是每次都把窗口右边多余的部分再贴到窗口坐标来,以后有机会我会作为一节笔记具体讲)同样的,我们贴出详细注释好的代码:#include &stdafx.h&
#include &stdio.h&
//定义结构体
struct BULLET
//bullet结构体代表飞机子弹
//子弹坐标
//子弹是否存在
//全局变量声明
HINSTANCE hI
HBITMAP bg,ship,
//存储背景图,飞机图,子弹图
DWORD tPre,tN
x,y,nowX,nowY;
//x,y代表鼠标光标所在位置,nowX,nowY代表飞机坐标,也是贴图的位置
//w为滚动背景所要裁剪的区域宽度,bcount记录飞机现有子弹数目
//声明一个“bullet”类型的数组,用来存储飞机发出的子弹
//全局函数声明
MyRegisterClass(HINSTANCE hInstance);
InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
MyPaint(HDC hdc);
//****WinMain函数,程序入口点函数**************************************
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
lpCmdLine,
MyRegisterClass(hInstance);
if (!InitInstance (hInstance, nCmdShow))
return FALSE;
//消息循环
GetMessage(&msg,NULL,NULL,NULL);
//初始化msg
while( msg.message!=WM_QUIT )
if( PeekMessage( &msg, NULL, 0,0 ,PM_REMOVE) )
TranslateMessage( &msg );
DispatchMessage( &msg );
tNow = GetTickCount();
if(tNow-tPre &= 40)
MyPaint(hdc);
return msg.wP
//****设计一个窗口类,类似填空题,使用窗口结构体*********************
ATOM MyRegisterClass(HINSTANCE hInstance)
WNDCLASSEX
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style
= CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS;
wcex.lpfnWndProc = (WNDPROC)WndP
wcex.cbClsExtra
wcex.cbWndExtra
wcex.hInstance
wcex.hIcon
wcex.hCursor
wcex.hCursor
= LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = NULL;
wcex.lpszClassName = &canvas&;
wcex.hIconSm
return RegisterClassEx(&wcex);
//****初始化函数*************************************
// 1.设定飞机初始位置
// 2.设定鼠标光标位置及隐藏
// 3.限制鼠标光标移动区域
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
POINT pt,lt,
hInst = hI
hWnd = CreateWindow(&canvas&, &绘图窗口& , WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if (!hWnd)
return FALSE;
MoveWindow(hWnd,10,10,640,480,true);
ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);
hdc = GetDC(hWnd);
mdc = CreateCompatibleDC(hdc);
bufdc = CreateCompatibleDC(hdc);
bmp = CreateCompatibleBitmap(hdc,640,480);
SelectObject(mdc,bmp);
bg = (HBITMAP)LoadImage(NULL,&bg.bmp&,IMAGE_BITMAP,648,480,LR_LOADFROMFILE);
ship = (HBITMAP)LoadImage(NULL,&ship.bmp&,IMAGE_BITMAP,100,148,LR_LOADFROMFILE);
bullet = (HBITMAP)LoadImage(NULL,&bullet.bmp&,IMAGE_BITMAP,10,20,LR_LOADFROMFILE);
//设定鼠标光标的x,y值,并设定飞机贴图坐标的“nowX”和“nowY”的值为(300,300)
nowX = 300;
nowY = 300;
//设定光标位置
pt.x = 300;
pt.y = 300;
ClientToScreen(hWnd,&pt);
SetCursorPos(pt.x,pt.y);
ShowCursor(false);
//隐藏鼠标光标
//限制鼠标光标移动区域
GetClientRect(hWnd,&rect);
//取得窗口内部矩形
//将矩形左上点坐标存入lt中
lt.x = rect.
lt.y = rect.
//将矩形右下坐标存入rb中
rb.x = rect.
rb.y = rect.
//将lt和rb的窗口坐标转换为屏幕坐标
ClientToScreen(hWnd,&);
ClientToScreen(hWnd,&rb);
//以屏幕坐标重新设定矩形区域
rect.left = lt.x;
rect.top = lt.y;
rect.right = rb.x;
rect.bottom = rb.y;
//限制鼠标光标移动区域
ClipCursor(&rect);
MyPaint(hdc);
return TRUE;
//****自定义绘图函数*********************************
// 1.设定飞机坐标并进行贴图
// 2.设定所有子弹坐标并进行贴图
// 3.显示真正的鼠标光标所在坐标
void MyPaint(HDC hdc)
char str[20] = &&;
//贴上背景图
SelectObject(bufdc,bg);
BitBlt(mdc,0,0,w,480,bufdc,640-w,0,SRCCOPY);
BitBlt(mdc,w,0,640-w,480,bufdc,0,0,SRCCOPY);
//计算飞机的贴图坐标,设定每次进行飞机贴图时,其贴图坐标(nowX,nowY)会以10个单位慢慢向鼠标光标所在的目的点(x,y)接近,直到两个坐标相同为止
if(nowX & x)
nowX += 10;
if(nowX & x)
nowX -=10;
if(nowX & x)
if(nowY & y)
nowY += 10;
if(nowY & y)
nowY -= 10;
if(nowY & y)
//贴上飞机图
SelectObject(bufdc,ship);
BitBlt(mdc,nowX,nowY,100,74,bufdc,0,74,SRCAND);
BitBlt(mdc,nowX,nowY,100,74,bufdc,0,0,SRCPAINT);
//子弹的贴图,先判断子弹数目“bcount”的值是否为“0”。若不为0,则对子弹数组中各个还存在的子弹按照其所在的坐标(b[i].x,b[i].y)循环进行贴图操作
SelectObject(bufdc,bullet);
if(bcount!=0)
for(i=0;i&30;i++)
if(b[i].exist)
//贴上子弹图
BitBlt(mdc,b[i].x,b[i].y,10,10,bufdc,0,10,SRCAND);
BitBlt(mdc,b[i].x,b[i].y,10,10,bufdc,0,0,SRCPAINT);
//设置下一个子弹的坐标。子弹是又右向左发射的,因此,每次其X轴上的坐标值递减10个单位,这样贴图会产生往左移动的效果。而如果子弹下次的坐标已超出窗口的可见范围(h[i].x&0),那么子弹设为不存在,并将子弹总数bcount变量值减1.
b[i].x -= 10;
if(b[i].x & 0)
b[i].exist =
//显示鼠标坐标
sprintf(str,&鼠标X坐标为%d
TextOut(mdc,0,0,str,strlen(str));
sprintf(str,&鼠标Y坐标为%d
TextOut(mdc,0,20,str,strlen(str));
BitBlt(hdc,0,0,640,480,mdc,0,0,SRCCOPY);
tPre = GetTickCount();
if(w==640)
//****消息处理函数***********************************
// 1.处理WM_LBUTTONDOWN消息发射子弹
// 2.处理WM_MOUSEMOVE消息设定飞机贴图坐标
// 3.在窗口结束时恢复鼠标移动区域
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
switch (message)
case WM_KEYDOWN:
//按键按下消息
if(wParam==VK_ESCAPE)
//按下【Esc】键
PostQuitMessage(0);
case WM_LBUTTONDOWN:
//单击鼠标左键消息
for(i=0;i&30;i++)
if(!b[i].exist)
b[i].x = nowX;
//子弹x坐标
b[i].y = nowY + 30; //子弹y坐标
b[i].exist =
//累加子弹数目
case WM_MOUSEMOVE:
x = LOWORD(lParam);
//取得鼠标X坐标
if(x & 530)
//设置临界坐标
else if(x & 0)
y = HIWORD(lParam);
//取得鼠标y坐标
if(y & 380)
else if(y & 0)
case WM_DESTROY:
//窗口结束消息
ClipCursor(NULL);
//恢复鼠标移动区域
DeleteDC(mdc);
DeleteDC(bufdc);
DeleteObject(bg);
DeleteObject(bullet);
DeleteObject(ship);
ReleaseDC(hWnd,hdc);
PostQuitMessage(0);
//其他消息
return DefWindowProc(hWnd, message, wParam, lParam);
下面是这个例子的效果图:我们移动鼠标,小飞机会跟着鼠标的移动而移动;点击鼠标,小飞机就会发射出绿色的子弹来;按下Esc键,这个小游戏就退出了。当然,这个小游戏还有些的不足,比如小飞机对鼠标的跟随有小小的延迟,也没有进行WndProc函数中case分支的拆分(在这里感谢的提醒)。关于这个小游戏的改进,是以后我们需要去做的。比如后面我们会找机会用DirectX中的DirectInput函数(也是处理输出消息的函数)来写一下这个小游戏的升级版,无论是画面上还是实现效果上都将进行升级。在之前的笔记里有朋友(感谢的提醒)提到,用MFC和ATL共享的新类CImage进行贴图会更加简单和先进,在这里说明一下,在后面的笔记里面会专门花一节来讲CImage类,目前和之前的笔记还是采用传统的GDI进行贴图操作。笔记十三到这里就结束了。本节笔记的源代码请点击这里下载: &&感谢一直支持【Visual&C++】游戏开发笔记系列专栏的朋友们,也请大家继续关注我的专栏,我一有时间就会把自己的学习心得,觉得比较好的知识点写出来和大家一起分享。精通游戏开发的路还很长很长,非常希望能和大家一起交流,共同学习,共同进步。大家看过后觉得值得一看的话,可以顶一下这篇文章,让更多的朋友有机会看到它。如果文章中有什么疏漏的地方,也请大家指正。也希望大家可以多留言来和我探讨编程相关的问题。最后,谢谢你们一直的支持~~~The&end.
参考知识库
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:3978452次
积分:33159
积分:33159
排名:第96名
原创:122篇
转载:25篇
评论:8397条
&&& 《OpenCV3编程入门》
PS:本书版权已经成功输出到台湾
■ 毛星云,网络ID“浅墨,90后,热爱游戏开发、游戏引擎、计算机图形学、图像处理等技术,就职于腾讯互娱。
■ 微软最有价值专家
■ 中国2013年度十大杰出IT博客作者
■ CSDN 2012年博客大赛年度博客之星
■ CSDN 2012年度十大风云专栏作者
■ 常活跃于、等网络社区
■ 本科毕业于南京航空航天大学中国乌克兰航天联合培养班,获乌克兰国立航空航天大学与南京航空航天大学双学位
■ 现就读于南京航空航天大学航天学院(2013级硕士研究生),将于2016年三月毕业
■ 邮箱: (#换成@)
PS:平时精力有限,大家的邮件不一定都能回复,请见谅。
《逐梦旅程:Windows游戏编程之从零开始》
文章:15篇
阅读:255011
文章:18篇
阅读:1081990
文章:57篇
阅读:1888730
(2)(1)(2)(3)(1)(1)(1)(1)(3)(3)(5)(4)(5)(7)(2)(5)(3)(1)(4)(6)(7)(6)(4)(6)(1)(3)(7)(4)(8)(17)(32)win7下调整鼠标滑轮灵敏度的方法
win7下调整鼠标滑轮灵敏度的方法
  点评:新入手一个微软无线鼠标4000,之前用的是罗技的,新鼠标操作还不错就是感觉滑轮过于灵敏。该如何进行调整呢?下面就介绍一个十分简单的方法
  在Windows 7操作系统下,打开&控制面板 -& 硬件和声音 -& 设备和打印机 -& 鼠标&,或者直接在&开始&的搜索栏中搜索&鼠标&。
  打开了&鼠标&界面,选择&滑轮&选项卡,将垂直滚动由之前的&3&修改成&1&,点击确定即可。
  如果还想调整指针和双击速度等,可以在&鼠标键&和&指针&里面进行调整,知道符合你的操作习惯为止。
&&&主编推荐
H3C认证Java认证Oracle认证
基础英语软考英语项目管理英语职场英语
.NETPowerBuilderWeb开发游戏开发Perl
二级模拟试题一级模拟试题一级考试经验四级考试资料
港口与航道工程建设工程法规及相关知识建设工程经济考试大纲矿业工程市政公用工程通信与广电工程
操作系统汇编语言计算机系统结构人工智能数据库系统微机与接口
软件测试软件外包系统分析与建模敏捷开发
法律法规历年试题软考英语网络管理员系统架构设计师信息系统监理师
高级通信工程师考试大纲设备环境综合能力
路由技术网络存储无线网络网络设备
CPMP考试prince2认证项目范围管理项目配置管理项目管理案例项目经理项目干系人管理
Powerpoint教程WPS教程
电子政务客户关系管理首席信息官办公自动化大数据
职称考试题目
就业指导签约违约职业测评
招生信息考研政治
网络安全安全设置工具使用手机安全
3DMax教程Flash教程CorelDraw教程Director教程
Dreamwaver教程HTML教程网站策划网站运营Frontpage教程
生物识别传感器物联网传输层物联网前沿技术物联网案例分析
互联网电信IT业界IT生活
Java核心技术J2ME教程
Linux系统管理Linux编程Linux安全AIX教程
Windows系统管理Windows教程Windows网络管理Windows故障
组织运营财务资本
视频播放文件压缩杀毒软件输入法微博
数据库开发Sybase数据库Informix数据库
&&&&&&&&&&&&&&&
希赛网 版权所有 & &&503: Service Unavailable
503: Service Unavailable
访问太频繁了,服务器要炸。503: Service Unavailable
503: Service Unavailable
访问太频繁了,服务器要炸。}

我要回帖

更多关于 鼠标滑轮不能用 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信