鱼C论坛

 找回密码
 立即注册
查看: 78|回复: 1

win32滚动条刷新的到底是什么?

[复制链接]
发表于 2024-7-26 16:17:02 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#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[128];
    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[i].szLabel, 1024, &iTarget);
            TextOut(hdc, 0, cyChar * i, sysmetrics[i].szLabel, iTarget);

            StringCchLength(sysmetrics[i].szDesc, 1024, &iTarget);
            TextOut(hdc, 22 * cxCaps, cyChar * i, sysmetrics[i].szDesc, iTarget);

            auto TextModeTemp = GetTextAlign(hdc);

            SetTextAlign(hdc, TA_RIGHT | TA_TOP);

            StringCchPrintf(szBuffer, 10, TEXT("%d"), GetSystemMetrics(sysmetrics[i].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滚动条刷新的到底是什么?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 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 小助理,如未能正确解答您的问题,请继续追问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-9-8 07:53

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表