win32滚动条刷新的到底是什么?
#include <windows.h>#include"sysmets.h"
#include<strsafe.h>
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
{
static TCHAR szAppName[] = TEXT("MyWindows");
HWND hwnd;
MSG msg;
WNDCLASS wndclass;
wndclass.style = CS_HREDRAW | CS_VREDRAW;
wndclass.lpfnWndProc = WndProc;
wndclass.cbClsExtra = 0;
wndclass.cbWndExtra = 0;
wndclass.hInstance = hInstance;
wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
wndclass.lpszMenuName = NULL;
wndclass.lpszClassName = szAppName;
if (!RegisterClass(&wndclass))
{
MessageBox(NULL, TEXT("这个程序需要在 Windows NT 才能执行!"), szAppName, MB_ICONERROR);
return 0;
}
hwnd = CreateWindow(szAppName,
TEXT("hello"),
WS_OVERLAPPEDWINDOW | WS_VSCROLL,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
NULL,
NULL,
hInstance,
NULL);
ShowWindow(hwnd, iCmdShow);
UpdateWindow(hwnd);
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return msg.wParam;
}
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam)
{
HDC hdc;
PAINTSTRUCT ps;
TEXTMETRIC tm;
TCHAR szBuffer;
static int cxChar, cxCaps, cyChar;
static int cxClient, cyClient;//客户区宽度和高度
size_t iTarget;
switch (message)
{
case WM_CREATE:
hdc = GetDC(hwnd);
GetTextMetrics(hdc, &tm);
cxChar = tm.tmAveCharWidth;//字体字符平均宽度
//如果是等宽字体不改变,如果是变宽字体(大写字母是小写字母宽度的1.5倍)
cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2;
//获取字符高度
cyChar = tm.tmHeight + tm.tmExternalLeading;
//初始化滚动条
SetScrollRange(hwnd, SB_VERT, 0, NUMLINES - 1, FALSE);
SetScrollPos(hwnd, SB_VERT, 20, TRUE);//刷新的到底是什么?
ReleaseDC(hwnd, hdc);
return 0;
case WM_PAINT:
hdc = BeginPaint(hwnd, &ps);
for (size_t i = 0; i < NUMLINES; i++)
{
StringCchLength(sysmetrics.szLabel, 1024, &iTarget);
TextOut(hdc, 0, cyChar * i, sysmetrics.szLabel, iTarget);
StringCchLength(sysmetrics.szDesc, 1024, &iTarget);
TextOut(hdc, 22 * cxCaps, cyChar * i, sysmetrics.szDesc, iTarget);
auto TextModeTemp = GetTextAlign(hdc);
SetTextAlign(hdc, TA_RIGHT | TA_TOP);
StringCchPrintf(szBuffer, 10, TEXT("%d"), GetSystemMetrics(sysmetrics.iIndex));
StringCchLength(szBuffer, 10, &iTarget);
TextOut(hdc, 22 * cxCaps + 40 * cxChar, cyChar * i, szBuffer, iTarget);
SetTextAlign(hdc, TextModeTemp);
}
EndPaint(hwnd, &ps);
return 0;
case WM_VSCROLL:
hdc = GetDC(hwnd);
SetTextAlign(hdc, TA_TOP | TA_RIGHT);
switch (LOWORD(wparam))
{
case SB_LINEUP:
TextOut(hdc, cxClient - 10, 10, TEXT("向上滚动一行"), 6);
break;
case SB_LINEDOWN:
TextOut(hdc, cxClient - 10, 10, TEXT("向下滚动一行"), 6);
break;
case SB_PAGEUP:
TextOut(hdc, cxClient - 10, 10, TEXT("向上滚动一页"), 6);
break;
case SB_PAGEDOWN:
TextOut(hdc, cxClient - 10, 10, TEXT("向下滚动一页"), 6);
break;
case SB_THUMBTRACK:
TextOut(hdc, cxClient - 10, 10, TEXT("别抓着我不放"), 6);
break;
}
ReleaseDC(hwnd, hdc);
return 0;
case WM_SIZE:
hdc = GetDC(hwnd);
StringCchPrintf(szBuffer, 128, TEXT("客户区分辨率是:%d * %d px"), LOWORD(lparam), HIWORD(lparam));
StringCchLength(szBuffer, 128, &iTarget);
TextOut(hdc, 10, 10, szBuffer, iTarget);
cxClient = LOWORD(lparam);
cyClient = HIWORD(lparam);
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc(hwnd, message, wparam, lparam);
}上述代码中win32滚动条刷新的到底是什么? 在你的代码中,Win32的滚动条刷新主要涉及到 `WM_VSCROLL` 消息的处理。这个消息是Windows发送给窗口过程的,用来通知窗口垂直滚动条的状态变化,比如用户向上向下滚动、抓住滑块移动等操作。
我们来一点点解析:
1. **初始化滚动条**:在 `WM_CREATE` 中,你使用 `SetScrollRange` 和 `SetScrollPos` 来设置滚动条的范围和当前位置。这相当于在新建窗口时设定了整个家庭的“上下左右”规则。
SetScrollRange(hwnd, SB_VERT, 0, NUMLINES - 1, FALSE);
SetScrollPos(hwnd, SB_VERT, 20, TRUE);
这里的 `NUMLINES` 是指你要显示的行数,而 `SB_VERT` 则说明这是一个垂直滚动条的设置。`20` 表示当前的位置,你可以想象一下这是你站在一座20层楼的摩天大楼前准备按电梯按钮,哦不,是滑动滚动条。
2. **处理滚动事件**:在 `WndProc` 函数中的 `WM_VSCROLL` 部分,你捕捉到了滚动条的各种操作,并且通过 `TextOut()` 将相应的文本显示在窗口的客户区。这些操作包括向上、向下滚动一行或一页,以及拖动滑块。
switch (LOWORD(wparam)) {
case SB_LINEUP:
TextOut(hdc, cxClient - 10, 10, TEXT("向上滚动一行"), 6);
break;
// 其他 cases...
}
每当用户采取行动时(比如,上滑一行),你就会用文字反馈给他们,像是个热心的邻居,总是及时告诉他们“嘿,你刚刚向上了!”
3. **刷新内容**:虽然在你的代码中没有显式调用重绘(如 `InvalidateRect()`),但实际上,在处理滚动条的时候,你可以通过改变当前滚动位置的方式来触发 `WM_PAINT` 消息,以便重新绘制窗口,显示更新后的内容。
总结一下,滚动条的“刷新”就是根据用户的操作(如滚动)来动态更新窗口的显示信息。在这个过程中,程序不断接收输入并对其做出反应,活像一场舞蹈比赛,精彩而又富有节奏感!
所以,可以说滚动条刷新的“到底是什么”,就是根据滚动事件,更新和重绘窗口内容,使用户体验更加流畅,就像你下班后在沙发上轻松回顾一天的工作一样舒适!
以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页:
[1]