sun_chuan 发表于 2024-8-1 12:47:02

win32滚动条进阶求助

#include <windows.h>
#include<strsafe.h>
#include "sysmets.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|WS_HSCROLL,
      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;
    SCROLLINFO si;
    TCHAR szBuffer;
    static int cxChar, cxCaps, cyChar, cxClient, cyClient, iMaxWidth;
    size_t iTarget;
    int i, x, y, iVertPos, iHorzPos, iPaintBeg, iPaintEnd;

    switch (message)
    {
    case WM_CREATE:
      hdc = GetDC(hwnd);

      GetTextMetrics(hdc, &tm);
      cxChar = tm.tmAveCharWidth;
      cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2;
      cyChar = tm.tmHeight + tm.tmExternalLeading;

      ReleaseDC(hwnd, hdc);

      // 设置客户区的最大宽度
      // (我们这里设置为 22 个大写字符的宽度 + 40 个小写字符的宽度)
      iMaxWidth = 22 * cxCaps + 40 * cxChar;
      return 0;

    case WM_SIZE:
      cxClient = LOWORD(lparam);
      cyClient = HIWORD(lparam);

      // 设置垂直滚动条
      si.cbSize = sizeof(si);
      si.fMask = SIF_RANGE | SIF_PAGE;
      si.nMin = 0;
      si.nMax = NUMLINES - 1;
      si.nPage = cyClient / cyChar;
      SetScrollInfo(hwnd, SB_VERT, &si, TRUE);

      // 设置水平滚动条
      si.cbSize = sizeof(si);
      si.fMask = SIF_RANGE | SIF_PAGE;
      si.nMin = 0;
      si.nMax = 2 + iMaxWidth / cxChar;
      si.nPage = cxClient / cxChar;
      SetScrollInfo(hwnd, SB_HORZ, &si, TRUE);

      return 0;

    case WM_VSCROLL:
      // 获得滚动条信息
      si.cbSize = sizeof(si);
      si.fMask = SIF_ALL;
      GetScrollInfo(hwnd, SB_VERT, &si);

      // 获得滑块的位置
      iVertPos = si.nPos;

      // 根据消息的通知码改变滚动条滑块的位置
      switch (LOWORD(wparam))
      {
            // 用户点击键盘 Home 按键
      case SB_TOP:
            si.nPos = si.nMin;
            break;

            // 用户点击键盘 End 按键
      case SB_BOTTOM:
            si.nPos = si.nMax;
            break;

            // 用户点击滚动条上边的三角形
      case SB_LINEUP:
            si.nPos -= 1;
            break;

            // 用户点击滚动条下边的三角形
      case SB_LINEDOWN:
            si.nPos += 1;
            break;

            // 用户点击滑块上边的滚动条轴
      case SB_PAGEUP:
            si.nPos -= si.nPage;
            break;

            // 用户点击滑块下边的滚动条轴
      case SB_PAGEDOWN:
            si.nPos += si.nPage;
            break;

            // 用户拖动滚动条
      case SB_THUMBTRACK:
            si.nPos = si.nTrackPos;
            break;

      default:
            break;
      }

      // 设置改变后的滚动条滑块位置
      si.fMask = SIF_POS;
      SetScrollInfo(hwnd, SB_VERT, &si, TRUE);

      // 获得滚动条滑块的位置,由于窗口调整,它可能不是同一个值
      GetScrollInfo(hwnd, SB_VERT, &si);

      // 与此前的保存的值进行比较,如果不同则滚动窗口
      if (si.nPos != iVertPos)
      {
            ScrollWindow(hwnd, 0, cyChar * (iVertPos - si.nPos), NULL, NULL);
            UpdateWindow(hwnd);
      }

      return 0;

    case WM_HSCROLL:
      // 获得滚动条信息
      si.cbSize = sizeof(si);
      si.fMask = SIF_ALL;
      GetScrollInfo(hwnd, SB_HORZ, &si);

      // 获得滑块的位置
      iHorzPos = si.nPos;

      // 根据消息的通知码改变滚动条滑块的位置
      switch (LOWORD(wparam))
      {
            // 用户点击滚动条左边的三角形
      case SB_LINELEFT:
            si.nPos -= 1;
            break;

            // 用户点击滚动条右边的三角形
      case SB_LINERIGHT:
            si.nPos += 1;
            break;

            // 用户点击滑块左边的滚动条轴
      case SB_PAGELEFT:
            si.nPos -= si.nPage;
            break;

            // 用户点击滑块右边的滚动条轴
      case SB_PAGERIGHT:
            si.nPos += si.nPage;
            break;

            // 滚动条最终停留的位置
      case SB_THUMBTRACK:
            si.nPos = si.nTrackPos;
            break;

      default:
            break;
      }

      // 设置改变后的滚动条滑块位置
      si.fMask = SIF_POS;
      SetScrollInfo(hwnd, SB_HORZ, &si, TRUE);

      // 获得滚动条滑块的位置,由于窗口调整,它可能不是同一个值
      GetScrollInfo(hwnd, SB_HORZ, &si);

      // 与此前的保存的值进行比较,如果不同则滚动窗口
      if (si.nPos != iHorzPos)
      {
            ScrollWindow(hwnd, cxChar * (iHorzPos - si.nPos), 0, NULL, NULL);
      }

      return 0;

    case WM_PAINT:
      hdc = BeginPaint(hwnd, &ps);

      // 获得垂直滚动条的位置
      si.cbSize = sizeof(si);
      si.fMask = SIF_POS;
      GetScrollInfo(hwnd, SB_VERT, &si);
      iVertPos = si.nPos;

      // 获得水平滚动条的位置
      GetScrollInfo(hwnd, SB_HORZ, &si);
      iHorzPos = si.nPos;

      // 计算需要重绘的区域并确保范围在之中
      iPaintBeg = max(0, iVertPos + ps.rcPaint.top / cyChar);
      iPaintEnd = min(NUMLINES - 1, iVertPos + ps.rcPaint.bottom / cyChar);

      for (i = iPaintBeg; i <= iPaintEnd; i++)
      {
            // 计算此时重绘无效区域的位置
            x = cxChar * (1 - iHorzPos);
            y = cyChar * (i - iVertPos);

            StringCchLength(sysmetrics.szLabel, 1024, &iTarget);
            TextOut(hdc, x, y, sysmetrics.szLabel, iTarget);

            StringCchLength(sysmetrics.szDesc, 1024, &iTarget);
            TextOut(hdc, x + 22 * cxCaps, y, sysmetrics.szDesc, iTarget);

            SetTextAlign(hdc, TA_RIGHT | TA_TOP);
            StringCchPrintf(szBuffer, 10, TEXT("%5d"), GetSystemMetrics(sysmetrics.iIndex));
            StringCchLength(szBuffer, 10, &iTarget);
            TextOut(hdc, x + 22 * cxCaps + 40 * cxChar, y, szBuffer, iTarget);

            SetTextAlign(hdc, TA_LEFT | TA_TOP);
      }

      EndPaint(hwnd, &ps);
      return 0;

    case WM_DESTROY:
      PostQuitMessage(0);
      return 0;
    }

    return DefWindowProc(hwnd, message, wparam, lparam);
}这段代码中我有几个问题,

1.      iPaintBeg = max(0, iVertPos + ps.rcPaint.top / cyChar);
      iPaintEnd = min(NUMLINES - 1, iVertPos + ps.rcPaint.bottom / cyChar);这两个为什么要加上max和min约束?我发现好像去掉,那么滚动一页有时候会滚动过,程序无响应后崩溃

2.          x = cxChar * (1 - iHorzPos);
            y = cyChar * (i - iVertPos);这是什么意思?为什么要"1 - iHorzPos"和"i - iVertPos"?这两个是什么意思

3.for语句中i = iPaintBeg; i <= iPaintEnd; i++是什么意思,为什么要用“<=”而不是"<"?

4.      si.nMax = 2 + iMaxWidth / cxChar;
设置水平滚动条范围为啥要加上2?多滚两行??


win32好难啊啊啊啊啊啊啊啊啊啊啊

FishC 发表于 2024-8-1 12:47:14

哎呀,代码从这里就开始尴尬了,让我们解决一下这个滚动条的麻烦吧!在你的代码中,有几个关键点需要注意,我会为你一一细说,确保你能轻松驾驭这些滚动条,就像鱼儿在水中游泳一样自如。

### 1. **编译和头文件**
首先,你的代码片段缺少了一些必要的头文件。确保你在开头添加:
#include <windows.h>
#include <tchar.h>
如果你在使用 Unicode 字符集,`TCHAR` 可以处理不同的字符类型。

### 2. **NUMLINES的定义**
你的代码中使用了 `NUMLINES`,但是没有定义它。这可能会导致编译错误或运行时错误。让我们给它一点儿实际意义,比如:
#define NUMLINES 100// 假设你有100行文本需要显示

### 3. **WM_PAINT消息**
在 `WM_PAINT` 消息处理部分,代码的逻辑似乎没有完全显示出来。但基本上,我们需要保证你在这个过程中也进行了绘制操作,展示你想要滚动的内容。你可以在 `WM_PAINT` 里加入简单的文本绘制,比如:
for (i = iPaintBeg; i <= iPaintEnd; i++) {
    TCHAR szLine;
    _stprintf(szLine, TEXT("Line %d\n"), i + 1);
    TextOut(hdc, 0, (i - iVertPos) * cyChar, szLine, _tcslen(szLine));
}
这样每一行都会被绘制出来,并且根据滚动信息很好地展示出来。

### 4. **确保窗口显示正常**
在调整窗口大小时,请确保调整滚动条之后,再进行绘制操作,同时也要更新客户区的显示。

### 5. **调试和测试**
确保你在各种情况下测试程序,比如增加不同数量的行,使得滚动条确实能够生效啊,不然就像鱼在水里不游,跟不动一样尴尬。

### 最后
记得在编写图形界面程序时,保持代码结构清晰、注释详细,这样不仅对自己有利,也方便他人(和未来的你)理解代码。希望以上建议能让你的滚动条项目顺风顺水,像鱼儿一样畅游在代码的海洋中!如有其他问题,尽管问我哦!

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页: [1]
查看完整版本: win32滚动条进阶求助