我正在写一个类似于" Paint"的程序。在Windows中。起初我试图做一个"铅笔"函数处理WM_MOUSEMOVE消息并正确调用SetPixel()。但是当鼠标移动太快时,并非所有像素都出现(它们看起来像稀疏)。我想我必须用另一个代码替换那个SetPixel()函数,但我不知道是什么。
答案 0 :(得分:5)
这是因为鼠标可能会在每次更新时移动多个像素,并且不是SetPixel中的错误。相反,你应该记住鼠标悬停的最后一个像素,并在两个像素之间画一条线(我认为正确的函数是DrawLine()
)。有关该方法的详细说明,请参阅ikh的答案。
答案 1 :(得分:3)
在窗口上抓住WM_LBUTTONDOWN
和set capture,保存第一个鼠标坐标。
抓住WM_MOUSEMOVE
,然后从第一个coord到现在coord绘制 line 。然后,将now coord保存到第一个鼠标坐标。
重复2。
抓住WM_LBUTTONUP
和release capture。
示例:
#include <windows.h>
#include <windowsx.h>
LRESULT CALLBACK WndProc(HWND hWnd, UINT iMsg, WPARAM wParam, LPARAM lParam);
int APIENTRY WinMain(HINSTANCE hInst, HINSTANCE, LPSTR, int)
{
WNDCLASS wc;
HWND hWnd;
MSG msg;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInst;
wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
wc.lpszMenuName = NULL;
wc.lpszClassName = L"adf";
wc.style = CS_HREDRAW | CS_VREDRAW;
wc.lpfnWndProc = WndProc;
RegisterClass(&wc);
hWnd = CreateWindow(L"adf", NULL, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hInst, NULL);
ShowWindow(hWnd, SW_NORMAL);
while (GetMessage(&msg, 0, 0, NULL))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return (int)msg.wParam;
}
LRESULT CALLBACK WndProc(HWND hWnd, UINT iMsg, WPARAM wParam, LPARAM lParam)
{
static int prevx, prevy;
switch (iMsg)
{
case WM_LBUTTONDOWN:
prevx = GET_X_LPARAM(lParam);
prevy = GET_Y_LPARAM(lParam);
SetCapture(hWnd);
return 0;
case WM_LBUTTONUP:
ReleaseCapture();
return 0;
case WM_MOUSEMOVE:
if (GetCapture() == hWnd)
{
HDC hdc = GetDC(hWnd);
MoveToEx(hdc, prevx, prevy, NULL);
LineTo(hdc, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam));
prevx = GET_X_LPARAM(lParam);
prevy = GET_Y_LPARAM(lParam);
ReleaseDC(hWnd, hdc);
}
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc(hWnd, iMsg, wParam, lParam);
}