鱼C论坛

 找回密码
 立即注册
查看: 3657|回复: 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[i].szLabel, 1024, &iTarget);
                        TextOut(hdc, x, y, sysmetrics[i].szLabel, iTarget);

                        StringCchLength(sysmetrics[i].szDesc, 1024, &iTarget);
                        TextOut(hdc,x+ 22 * cxCaps,y, sysmetrics[i].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);
}


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

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

使用道具 举报

发表于 2014-10-23 11:28:03 | 显示全部楼层
请参考下,问题在响应滚动条消息后的矫正。
/* -------------------------------------------------------------------
MyWindows.c -- 基本窗口模型
《Windows 程序设计(SDK)》视频教程
--------------------------------------------------------------------*/

#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("鱼C工作室"),
                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[10];
        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_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 != 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;

                // 计算需要重绘的区域并确保范围在[0, NUMLINES-1]之中
                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[i].szLabel, 1024, &iTarget);
                        TextOut(hdc, x, y, sysmetrics[i].szLabel, iTarget);

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

                        SetTextAlign(hdc, TA_RIGHT | TA_TOP);
                        StringCchPrintf(szBuffer, 10, TEXT("%5d"), GetSystemMetrics(sysmetrics[i].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);
}
想知道小甲鱼最近在做啥?请访问 -> 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-11-22 18:59

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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