sun_chuan 发表于 2024-7-26 16:17:02

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滚动条刷新的到底是什么?

FishC 发表于 2024-7-26 16:18:16

在你的代码中,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]
查看完整版本: win32滚动条刷新的到底是什么?