鱼C论坛

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

win32滚动条进阶求助

[复制链接]
发表于 2024-8-1 12:47:02 | 显示全部楼层 |阅读模式

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

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

x
  1. #include <windows.h>
  2. #include<strsafe.h>
  3. #include "sysmets.h"
  4. LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

  5. int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
  6. {
  7.     static TCHAR szAppName[] = TEXT("MyWindows");
  8.     HWND hwnd;
  9.     MSG msg;
  10.     WNDCLASS wndclass;

  11.     wndclass.style = CS_HREDRAW | CS_VREDRAW;
  12.     wndclass.lpfnWndProc = WndProc;
  13.     wndclass.cbClsExtra = 0;
  14.     wndclass.cbWndExtra = 0;
  15.     wndclass.hInstance = hInstance;
  16.     wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  17.     wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
  18.     wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
  19.     wndclass.lpszMenuName = NULL;
  20.     wndclass.lpszClassName = szAppName;

  21.     if (!RegisterClass(&wndclass))
  22.     {
  23.         MessageBox(NULL, TEXT("这个程序需要在 Windows NT 才能执行!"), szAppName, MB_ICONERROR);
  24.         return 0;
  25.     }

  26.     hwnd = CreateWindow(szAppName,
  27.         TEXT("hello"),
  28.         WS_OVERLAPPEDWINDOW | WS_VSCROLL|WS_HSCROLL,
  29.         CW_USEDEFAULT,
  30.         CW_USEDEFAULT,
  31.         CW_USEDEFAULT,
  32.         CW_USEDEFAULT,
  33.         NULL,
  34.         NULL,
  35.         hInstance,
  36.         NULL);

  37.     ShowWindow(hwnd, iCmdShow);
  38.     UpdateWindow(hwnd);

  39.     while (GetMessage(&msg, NULL, 0, 0))
  40.     {
  41.         TranslateMessage(&msg);
  42.         DispatchMessage(&msg);
  43.     }

  44.     return msg.wParam;
  45. }

  46. LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam)
  47. {
  48.     HDC hdc;
  49.     PAINTSTRUCT ps;
  50.     TEXTMETRIC tm;
  51.     SCROLLINFO si;
  52.     TCHAR szBuffer[10];
  53.     static int cxChar, cxCaps, cyChar, cxClient, cyClient, iMaxWidth;
  54.     size_t iTarget;
  55.     int i, x, y, iVertPos, iHorzPos, iPaintBeg, iPaintEnd;

  56.     switch (message)
  57.     {
  58.     case WM_CREATE:
  59.         hdc = GetDC(hwnd);

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

  64.         ReleaseDC(hwnd, hdc);

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

  69.     case WM_SIZE:
  70.         cxClient = LOWORD(lparam);
  71.         cyClient = HIWORD(lparam);

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

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

  86.         return 0;

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

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

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

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

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

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

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

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

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

  125.         default:
  126.             break;
  127.         }

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

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

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

  139.         return 0;

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

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

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

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

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

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

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

  170.         default:
  171.             break;
  172.         }

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

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

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

  183.         return 0;

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

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

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

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

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

  202.             StringCchLength(sysmetrics[i].szLabel, 1024, &iTarget);
  203.             TextOut(hdc, x, y, sysmetrics[i].szLabel, iTarget);

  204.             StringCchLength(sysmetrics[i].szDesc, 1024, &iTarget);
  205.             TextOut(hdc, x + 22 * cxCaps, y, sysmetrics[i].szDesc, iTarget);

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

  210.             SetTextAlign(hdc, TA_LEFT | TA_TOP);
  211.         }

  212.         EndPaint(hwnd, &ps);
  213.         return 0;

  214.     case WM_DESTROY:
  215.         PostQuitMessage(0);
  216.         return 0;
  217.     }

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

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好难啊啊啊啊啊啊啊啊啊啊啊
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2024-8-1 12:47:14 | 显示全部楼层
哎呀,代码从这里就开始尴尬了,让我们解决一下这个滚动条的麻烦吧!在你的代码中,有几个关键点需要注意,我会为你一一细说,确保你能轻松驾驭这些滚动条,就像鱼儿在水中游泳一样自如。

### 1. **编译和头文件**
首先,你的代码片段缺少了一些必要的头文件。确保你在开头添加:
  1. #include <windows.h>
  2. #include <tchar.h>
复制代码

如果你在使用 Unicode 字符集,`TCHAR` 可以处理不同的字符类型。

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


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

这样每一行都会被绘制出来,并且根据滚动信息很好地展示出来。

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

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

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

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-6-6 13:52

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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