这些都是很实用的提示。我们这里实现鼠标移动时实时提示当前光标所在的客户区坐标值。效果图如下:
【实时提示光标的坐标值的提示框】
从图中我们可以看到,客户区是很多的小格子,这个就是坐标线,按照每10个单位刻度画一条线。当然,我们提示坐标,并不需要依赖这个格子,在哪个位置都可以知道坐标,这样用来装逼,显得高大上点,哈哈哈。其实就是用两个循环划线就可以了。这个在WM_PAINT消息中绘制即可,代码如下:GetClientRect(hwnd,&rect);
SelectObject(hdc,CreatePen(PS_SOLID,1,RGB(120,120,120)));
for (int i=0;i<rect.right;i++)
{
MoveToEx(hdc,i*10,0,NULL);
LineTo(hdc,i*10,rect.bottom);
}
for (int i=0;i<rect.bottom;i++)
{
MoveToEx(hdc,0,i*10,NULL);
LineTo(hdc,rect.right,i*10);
}
代码很简单,就不详细解释了。然后我们鼠标在客户区移动时就可以看到提示框。所以我们要处理WM_MOUSEMOVE消息,在这个消息实时处理移动时提示当前光标的坐标。
那么我们用什么作为显示提示文字的载体呢?有点经验的就知道用窗口咯。我创建一个无边框的扁平的小窗口即可,这就要用到CreateWindow函数。然后在移动光标时,我们只需要移动窗口即可跟着光标跑了。移动窗口就要用MoveWindow函数。然后要显示窗口中的文字,我们使用SetWindowText就可以设置窗口的文字。函数的使用请查MSDN。
在移动光标也会,你可能会发现提示文字的小窗口上面,竟然有一个虚框,显得很丑。出现这个问题,是因为客户区的焦点丢失了,跑到了提示窗口上了。
所以,我们处理客户区的焦点丢失消息WM_KILLFOCUS,在丢失的时候,又重新设置焦点到客户区,这样提示窗口就始终不会得到焦点,所以就不会出现虚框了。
实现的完整代码如下:
#include "windows.h"
#include <Windowsx.h>//GET_X_LPARAM,GET_Y_LPARAM宏需要这个头文件
#include <tchar.h>
// - 项目是Unicode字符集
HINSTANCE g_hInstance=NULL;
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
HDC hdc;
PAINTSTRUCT ps;
TCHAR Info[100]=_T("C++技术网http://www.cjjjs.com");
RECT rect;
static HWND hTip=NULL;
switch (message)
{
case WM_PAINT:
hdc = BeginPaint(hwnd,&ps);
GetClientRect(hwnd,&rect);
SelectObject(hdc,CreatePen(PS_SOLID,1,RGB(120,120,120)));
for (int i=0;i<rect.right;i++)
{
MoveToEx(hdc,i*10,0,NULL);
LineTo(hdc,i*10,rect.bottom);
}
for (int i=0;i<rect.bottom;i++)
{
MoveToEx(hdc,0,i*10,NULL);
LineTo(hdc,rect.right,i*10);
}
EndPaint(hwnd,&ps);
return 0;
case WM_KILLFOCUS:
SetFocus(hwnd);
return 0;
case WM_MOUSEMOVE:
if (!hTip)
{
hTip = CreateWindow(_T("Button"),_T("测试"),BS_FLAT|WS_POPUP,GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam)+50,100,40,hwnd,NULL,g_hInstance,NULL);
ShowWindow(hTip,SW_SHOWNORMAL);
}
else
{
MoveWindow(hTip,GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam)+50,100,40,TRUE);
TCHAR msg[100]=_T("");
wsprintf(msg,_T("(%d ,%d)"),GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam));
SetWindowText(hTip,msg);
}
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
default:
return DefWindowProc(hwnd, message, wParam, lParam);
}
}
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
TCHAR ClassName[] = _T("MyClass");
TCHAR title1[] = _T("C++技术网http://www.cjjjs.com");
WNDCLASS wndClass;
wndClass.cbClsExtra=0;
wndClass.cbWndExtra=0;
wndClass.hbrBackground= (HBRUSH)GetStockObject(WHITE_BRUSH);
wndClass.hCursor=LoadCursor(NULL,IDC_ARROW);
wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
wndClass.hInstance = hInstance;
wndClass.lpfnWndProc = WinProc;
wndClass.lpszClassName = ClassName;
wndClass.lpszMenuName=NULL;
wndClass.style=CS_HREDRAW|CS_VREDRAW;
if(!RegisterClass(&wndClass))return 0;
g_hInstance = hInstance;
HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,0,0,440,400,NULL,NULL,hInstance,NULL);
ShowWindow(hwnd,SW_SHOWNORMAL);
MSG msg;
while (GetMessage(&msg,NULL,0,0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return msg.wParam;
}