鱼C论坛

 找回密码
 立即注册
查看: 3335|回复: 4

关于15课滚动条的疑问(附源码)!!!

[复制链接]
发表于 2014-10-15 23:09:41 | 显示全部楼层 |阅读模式

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

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

x
本帖最后由 zfycike 于 2014-10-15 23:10 编辑

LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
        HDC hdc;
        PAINTSTRUCT ps;
        TEXTMETRIC tm;
        SCROLLINFO si;
        TCHAR szBuffer[10];
        static int cxClient;    //客户区的宽度
        static int cyClient;    //客户区的高度
        static int cxClientMax; //客户区内容显示的最大宽度(超过此宽度则加入滚动条)
        
        static int cxChar;      //字体的平均水平宽度
        static int cyChar;      //字体的垂直高度
        static int cxCaps;      //字体的大写字符的水平宽度

        static int sbxPos;      //当前水平滚动条滑块的位置
        static int sbyPos;      //当前垂直滚动条滑块的位置

        int i;                  //循环计数器
        int x, y;               //水平和垂直的坐标

        int FirstLine;          //失效区域的第一行(需重绘的第一行)
        int LastLine;           //失效区域的最后一行(需重绘的最后一行)

        //HRESULT hr;
    size_t iTarget;          //用于存放字符数组的长度

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

                GetTextMetrics(hdc, &tm);
                cxChar = tm.tmAveCharWidth;//字体中小写字符的平均宽度(一般定义为字母x的宽度)
                cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2;//低位0表示等宽,低位1变宽,大写字母
                                                                        //是小写的3/2倍
                cyChar = tm.tmHeight;            //字符高度+两行之间的间距
               
                ReleaseDC(hwnd, hdc);
        //        SetScrollRange(hwnd,SB_VERT,0,NUMLINES-1,FALSE);
        //        SetScrollPos(hwnd,SB_VERT,0,TRUE);
                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 + cxClient / cxChar+70;
                si.nPage = cxClient / cxChar;
                SetScrollInfo(hwnd,SB_HORZ,&si,TRUE);

                return 0;

        case WM_HSCROLL:
                //获得水平滚动条的所有信息
                si.cbSize = sizeof(si);
                si.fMask = SIF_ALL;
                GetScrollInfo(hwnd,SB_HORZ,&si);
               
                //保存当前滑块位置,迟些进行比较
                sbxPos = 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;
                case SB_THUMBPOSITION:
                        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 != sbxPos)
                {
                        ScrollWindow(hwnd,cxChar*(sbxPos-si.nPos),0,NULL,NULL);
                        //UpdateWindow(hwnd);
                }

                return 0;

        case WM_VSCROLL:
                hdc = GetDC(hwnd);
                SetTextAlign(hdc,TA_TOP|TA_RIGHT);

                //获得垂直滚动条的所有信息
                si.cbSize = sizeof(si);
                si.fMask = SIF_ALL;
                GetScrollInfo(hwnd,SB_VERT,&si);

                //保存当前滑块位置,迟些进行比较
                sbyPos = 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 -= cyClient / cyChar;
                        break;

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

                case SB_THUMBTRACK:
                        si.nPos = si.nTrackPos;
                        break;

                default:
                        break;

        //        case SB_THUMBPOSITION:
        //                si.nPos = HIWORD(wParam);
        //                break;
                }

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

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

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

        //        si.nPos = max(0, min(iVscrollPos, NUMLINES - 1));
        
               
                ReleaseDC(hwnd, hdc);
                return 0;

        case WM_PAINT:
                //准备绘制
                hdc = BeginPaint(hwnd, &ps);

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

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

                //计算需要重绘的区域
                FirstLine = max(0,sbyPos + ps.rcPaint.top/ cyChar);
                LastLine = min(NUMLINES -1 ,sbyPos + ps.rcPaint.bottom / cyChar);

                for (i = FirstLine; i < LastLine; i++)
                {
                        //si.cbSize = sizeof(si);
                        //si.fMask = SIF_POS;

                        //GetScrollInfo(hwnd,SB_VERT,&si);

                        x = cxChar * (1 - sbxPos);
                        y = cyChar*(i - sbyPos);
                        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);
}


以上是源码
运行时,点击垂直滚动条的下面箭头或者拖动滚动条,下面的内容显示时会出现判断重绘区域错误的问题,计算的时候会出现显示不完整,调整窗口高度,微调,会找到一个合适高度,就不会出现显示不完整了.找了好久,没发现问题出在什么地方.求大家帮帮忙.

Windows 8 x64-2014-10-15-23-10-27.png
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2014-10-23 11:28:03 | 显示全部楼层
请参考下,问题在响应滚动条消息后的矫正。

  1. /* -------------------------------------------------------------------
  2. MyWindows.c -- 基本窗口模型
  3. 《Windows 程序设计(SDK)》视频教程
  4. --------------------------------------------------------------------*/

  5. #include <windows.h>
  6. #include "strsafe.h"
  7. #include "sysmets.h"

  8. LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

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

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

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

  30.         hwnd = CreateWindow(szAppName,
  31.                 TEXT("鱼C工作室"),
  32.                 WS_OVERLAPPEDWINDOW | WS_VSCROLL | WS_HSCROLL,
  33.                 CW_USEDEFAULT,
  34.                 CW_USEDEFAULT,
  35.                 CW_USEDEFAULT,
  36.                 CW_USEDEFAULT,
  37.                 NULL,
  38.                 NULL,
  39.                 hInstance,
  40.                 NULL);

  41.         ShowWindow(hwnd, iCmdShow);
  42.         UpdateWindow(hwnd);

  43.         while (GetMessage(&msg, NULL, 0, 0))
  44.         {
  45.                 TranslateMessage(&msg);
  46.                 DispatchMessage(&msg);
  47.         }

  48.         return msg.wParam;
  49. }

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

  60.         switch (message)
  61.         {
  62.         case WM_CREATE:
  63.                 hdc = GetDC(hwnd);

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

  68.                 ReleaseDC(hwnd, hdc);

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

  73.         case WM_SIZE:
  74.                 cxClient = LOWORD(lParam);
  75.                 cyClient = HIWORD(lParam);

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

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

  90.                 return 0;

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

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

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

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

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

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

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

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

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

  129.                 default:
  130.                         break;
  131.                 }

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

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

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

  143.                 return 0;

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

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

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

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

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

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

  170.                         // 滚动条最终停留的位置
  171.                 case SB_THUMBPOSITION:
  172.                         si.nPos = si.nTrackPos;
  173.                         break;

  174.                 default:
  175.                         break;
  176.                 }

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

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

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

  187.                 return 0;

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

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

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

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

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

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

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

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

  214.                         SetTextAlign(hdc, TA_LEFT | TA_TOP);
  215.                 }

  216.                 EndPaint(hwnd, &ps);
  217.                 return 0;

  218.         case WM_DESTROY:
  219.                 PostQuitMessage(0);
  220.                 return 0;
  221.         }

  222.         return DefWindowProc(hwnd, message, wParam, lParam);
  223. }
复制代码
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2014-10-25 22:34:51 | 显示全部楼层
谢谢小甲鱼~~~
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2015-8-27 08:23:32 | 显示全部楼层
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2016-1-7 13:45:43 | 显示全部楼层
for (i = FirstLine; i < LastLine; i++)
                             此处少个等号...
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2024-4-26 20:46

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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