本文的程序来自《windows程序设计(第五版)》
设计滚动条的初衷,是因为客户区需要显示的东西太多了,在这个头文件中,我们包含了一大堆的系统的信息:
//总行数
#define NUMLINES ((int) (sizeof sysmetrics / sizeof sysmetrics [0]))
struct
{
int Index ;
TCHAR* szLabel;
TCHAR* szDesc ;
}
//结构体数组
sysmetrics [] =
{
SM_CXSCREEN, TEXT ("SM_CXSCREEN"), TEXT ("Screen width in pixels"),
SM_CYSCREEN, TEXT ("SM_CYSCREEN"), TEXT ("Screen height in pixels"),
SM_CXVSCROLL, TEXT ("SM_CXVSCROLL"), TEXT ("Vertical scroll width"),
SM_CYHSCROLL, TEXT ("SM_CYHSCROLL"), TEXT ("Horizontal scroll height"),
SM_CYCAPTION, TEXT ("SM_CYCAPTION"), TEXT ("Caption bar height"),
SM_CXBORDER, TEXT ("SM_CXBORDER"), TEXT ("Window border width"),
SM_CYBORDER, TEXT ("SM_CYBORDER"), TEXT ("Window border height"),
SM_CXFIXEDFRAME, TEXT ("SM_CXFIXEDFRAME"), TEXT ("Dialog window frame width"),
SM_CYFIXEDFRAME, TEXT ("SM_CYFIXEDFRAME"), TEXT ("Dialog window frame height"),
SM_CYVTHUMB, TEXT ("SM_CYVTHUMB"), TEXT ("Vertical scroll thumb height"),
SM_CXHTHUMB, TEXT ("SM_CXHTHUMB"), TEXT ("Horizontal scroll thumb width"),
SM_CXICON, TEXT ("SM_CXICON"), TEXT ("Icon width"),
SM_CYICON, TEXT ("SM_CYICON"), TEXT ("Icon height"),
SM_CXCURSOR, TEXT ("SM_CXCURSOR"), TEXT ("Cursor width"),
SM_CYCURSOR, TEXT ("SM_CYCURSOR"), TEXT ("Cursor height"),
SM_CYMENU, TEXT ("SM_CYMENU"), TEXT ("Menu bar height"),
SM_CXFULLSCREEN, TEXT ("SM_CXFULLSCREEN"), TEXT ("Full screen client area width"),
SM_CYFULLSCREEN, TEXT ("SM_CYFULLSCREEN"), TEXT ("Full screen client area height"),
SM_CYKANJIWINDOW, TEXT ("SM_CYKANJIWINDOW"), TEXT ("Kanji window height"),
SM_MOUSEPRESENT, TEXT ("SM_MOUSEPRESENT"), TEXT ("Mouse present flag"),
SM_CYVSCROLL, TEXT ("SM_CYVSCROLL"), TEXT ("Vertical scroll arrow height"),
SM_CXHSCROLL, TEXT ("SM_CXHSCROLL"), TEXT ("Horizontal scroll arrow width"),
SM_DEBUG, TEXT ("SM_DEBUG"), TEXT ("Debug version flag"),
SM_SWAPBUTTON, TEXT ("SM_SWAPBUTTON"), TEXT ("Mouse buttons swapped flag"),
SM_CXMIN, TEXT ("SM_CXMIN"), TEXT ("Minimum window width"),
SM_CYMIN, TEXT ("SM_CYMIN"), TEXT ("Minimum window height"),
SM_CXSIZE, TEXT ("SM_CXSIZE"), TEXT ("Min/Max/Close button width"),
SM_CYSIZE, TEXT ("SM_CYSIZE"), TEXT ("Min/Max/Close button height"),
SM_CXSIZEFRAME, TEXT ("SM_CXSIZEFRAME"), TEXT ("Window sizing frame width"),
SM_CYSIZEFRAME, TEXT ("SM_CYSIZEFRAME"), TEXT ("Window sizing frame height"),
SM_CXMINTRACK, TEXT ("SM_CXMINTRACK"), TEXT ("Minimum window tracking width"),
SM_CYMINTRACK, TEXT ("SM_CYMINTRACK"), TEXT ("Minimum window tracking height"),
SM_CXDOUBLECLK, TEXT ("SM_CXDOUBLECLK"), TEXT ("Double click x tolerance"),
SM_CYDOUBLECLK, TEXT ("SM_CYDOUBLECLK"), TEXT ("Double click y tolerance"),
SM_CXICONSPACING, TEXT ("SM_CXICONSPACING"), TEXT ("Horizontal icon spacing"),
SM_CYICONSPACING, TEXT ("SM_CYICONSPACING"), TEXT ("Vertical icon spacing"),
SM_MENUDROPALIGNMENT, TEXT ("SM_MENUDROPALIGNMENT"), TEXT ("Left or right menu drop"),
SM_PENWINDOWS, TEXT ("SM_PENWINDOWS"), TEXT ("Pen extensions installed"),
SM_DBCSENABLED, TEXT ("SM_DBCSENABLED"), TEXT ("Double-Byte Char Set enabled"),
SM_CMOUSEBUTTONS, TEXT ("SM_CMOUSEBUTTONS"), TEXT ("Number of mouse buttons"),
SM_SECURE, TEXT ("SM_SECURE"), TEXT ("Security present flag"),
SM_CXEDGE, TEXT ("SM_CXEDGE"), TEXT ("3-D border width"),
SM_CYEDGE, TEXT ("SM_CYEDGE"), TEXT ("3-D border height"),
SM_CXMINSPACING, TEXT ("SM_CXMINSPACING"), TEXT ("Minimized window spacing width"),
SM_CYMINSPACING, TEXT ("SM_CYMINSPACING"), TEXT ("Minimized window spacing height"),
SM_CXSMICON, TEXT ("SM_CXSMICON"), TEXT ("Small icon width"),
SM_CYSMICON, TEXT ("SM_CYSMICON"), TEXT ("Small icon height"),
SM_CYSMCAPTION, TEXT ("SM_CYSMCAPTION"), TEXT ("Small caption height"),
SM_CXSMSIZE, TEXT ("SM_CXSMSIZE"), TEXT ("Small caption button width"),
SM_CYSMSIZE, TEXT ("SM_CYSMSIZE"), TEXT ("Small caption button height"),
SM_CXMENUSIZE, TEXT ("SM_CXMENUSIZE"), TEXT ("Menu bar button width"),
SM_CYMENUSIZE, TEXT ("SM_CYMENUSIZE"), TEXT ("Menu bar button height"),
SM_ARRANGE, TEXT ("SM_ARRANGE"), TEXT ("How minimized windows arranged"),
SM_CXMINIMIZED, TEXT ("SM_CXMINIMIZED"), TEXT ("Minimized window width"),
SM_CYMINIMIZED, TEXT ("SM_CYMINIMIZED"), TEXT ("Minimized window height"),
SM_CXMAXTRACK, TEXT ("SM_CXMAXTRACK"), TEXT ("Maximum draggable width"),
SM_CYMAXTRACK, TEXT ("SM_CYMAXTRACK"), TEXT ("Maximum draggable height"),
SM_CXMAXIMIZED, TEXT ("SM_CXMAXIMIZED"), TEXT ("Width of maximized window"),
SM_CYMAXIMIZED, TEXT ("SM_CYMAXIMIZED"), TEXT ("Height of maximized window"),
SM_NETWORK, TEXT ("SM_NETWORK"), TEXT ("Network present flag"),
SM_CLEANBOOT, TEXT ("SM_CLEANBOOT"), TEXT ("How system was booted"),
SM_CXDRAG, TEXT ("SM_CXDRAG"), TEXT ("Avoid drag x tolerance"),
SM_CYDRAG, TEXT ("SM_CYDRAG"), TEXT ("Avoid drag y tolerance"),
SM_SHOWSOUNDS, TEXT ("SM_SHOWSOUNDS"), TEXT ("Present sounds visually"),
SM_CXMENUCHECK, TEXT ("SM_CXMENUCHECK"), TEXT ("Menu check-mark width"),
SM_CYMENUCHECK, TEXT ("SM_CYMENUCHECK"), TEXT ("Menu check-mark height"),
SM_SLOWMACHINE, TEXT ("SM_SLOWMACHINE"), TEXT ("Slow processor flag"),
SM_MIDEASTENABLED, TEXT ("SM_MIDEASTENABLED"), TEXT ("Hebrew and Arabic enabled flag"),
SM_MOUSEWHEELPRESENT, TEXT ("SM_MOUSEWHEELPRESENT"), TEXT ("Mouse wheel present flag"),
SM_XVIRTUALSCREEN, TEXT ("SM_XVIRTUALSCREEN"), TEXT ("Virtual screen x origin"),
SM_YVIRTUALSCREEN, TEXT ("SM_YVIRTUALSCREEN"), TEXT ("Virtual screen y origin"),
SM_CXVIRTUALSCREEN, TEXT ("SM_CXVIRTUALSCREEN"), TEXT ("Virtual screen width"),
SM_CYVIRTUALSCREEN, TEXT ("SM_CYVIRTUALSCREEN"), TEXT ("Virtual screen height"),
SM_CMONITORS, TEXT ("SM_CMONITORS"), TEXT ("Number of monitors"),
SM_SAMEDISPLAYFORMAT, TEXT ("SM_SAMEDISPLAYFORMAT"), TEXT ("Same color format flag")
} ;
第一行定义的,是数组有多少个元素。之后我们定义了结构体数组。元素个数很多,一行显示不下,那么如何添加滚动条呢?
我们先看程序:
#include <windows.h>
#include "sysmets.h"
LRESULT CALLBACK WndProc (HWND,UINT,WPARAM,LPARAM);
int WINAPI WinMain(HINSTANCE hInstance, //当前实例句柄
HINSTANCE hPrevInstance, //先前实例句柄
LPSTR lpCmdLine, //命令行
int iCmdShow) //显示状态
{
static TCHAR szAppName[] = TEXT("显示系统内容");
//窗口句柄
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))
{
return -1;
}
//创建窗口
hwnd = CreateWindow(szAppName, //窗口类的名称,必须是已经注册的
TEXT("系统内容"), //窗口标题
WS_OVERLAPPEDWINDOW | WS_VSCROLL, //窗口风格,加入垂直滚动条
CW_USEDEFAULT, //X坐标
CW_USEDEFAULT, //Y坐标
CW_USEDEFAULT, //宽度
CW_USEDEFAULT, //高度
NULL, //父窗口句柄
NULL, //菜单窗口句柄
hInstance, //高级版本的windos忽略
NULL);
//显示窗口
//ShowWindow(hwnd,SW_SHOWNA);
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)
{
//字符的宽度,大写字母宽度,字符高度
static int cxChar, cxCaps, cyChar ;
//窗口大小
static int cxClient, cyClient ;
//滚动条位置
static int iVscrollPos;
HDC hdc;
//该变量用于索引sysmets.h中定义的结构体数组sysmetrics[]的每个元素
int i;
//输出文本的垂直位置
int y;
//绘图结构
PAINTSTRUCT ps;
//字符串
TCHAR szBuffer [10];
//字体信息结构
TEXTMETRIC tm;
switch(message)
{
case WM_CREATE:
hdc = GetDC(hwnd);
//取得内定系统字体的文字大小,存在放在tm里
GetTextMetrics (hdc, &tm);
//平均字符宽
cxChar = tm.tmAveCharWidth ;
//大写字母的平平均宽度
cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2 ;
//字符总高度:高度+行间距
cyChar = tm.tmHeight + tm.tmExternalLeading ;
ReleaseDC(hwnd,hdc);
//设置滚动条范围:参数为:窗口句柄,滚动条类型,最小位置,最大位置,是否重绘
SetScrollRange (hwnd, SB_VERT, 0, NUMLINES - 1, FALSE);
//滚动条的方块的位置,参数为:窗口句柄,滚动条类型,新的位置
SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE);
return 0;
case WM_SIZE:
cxClient = LOWORD (lParam) ;
cyClient = HIWORD (lParam) ;
return 0;
case WM_PAINT:
hdc = BeginPaint (hwnd, &ps) ;
for(i = 0;i < NUMLINES;i++ )
{
//行宽度*变动的行数
y = cyChar * (i - iVscrollPos) ;
//输出字符串到指定的区域(使用当前的字体,背景颜色,颜色)
//参数为:设备内容句柄,x坐标,y坐标,带输出的字符串(与是否/0结尾无关),字符的个数
//lstrlen计算字符串长度
TextOut(hdc,0, y,sysmetrics[i].szLabel,lstrlen(sysmetrics[i].szLabel));
//从22个大写字母以后的位置输出,因为第一列最多只有20个大写字母
TextOut(hdc,22*cxCaps, y,sysmetrics[i].szDesc,lstrlen(sysmetrics[i].szDesc));
//最后一列是右对齐的
SetTextAlign(hdc,TA_RIGHT | TA_TOP);
//40*cxChar是第二列及第三列的列宽
//wsprintf用来给指定的buffer中格式化并存储字符串(0结尾),返回值为字符串的数目
//GetSystemMetrics用来获取系统参数,返回值为系统参数
TextOut(hdc,22*cxCaps+40*cxChar,y,szBuffer,wsprintf(szBuffer,TEXT("%5d"),GetSystemMetrics (sysmetrics[i].Index)));
//再把对齐方式改回去,否则下一次循环时也是右对齐
SetTextAlign(hdc,TA_LEFT | TA_TOP);
}
EndPaint (hwnd, &ps) ;
return 0;
//滚动条消息
case WM_VSCROLL:
//通过滚动条消息的wParam表明滚动条的操作
switch(LOWORD (wParam))
{
//点一下上箭头:向下翻一行
case SB_LINEUP:
iVscrollPos -=1;
break;
//点一下下箭头:向上翻一行
case SB_LINEDOWN:
iVscrollPos +=1;
break;
//点一下方块上面:向上翻一页,每一页是:(客户区/一行高度)这么多行
case SB_PAGEUP:
iVscrollPos -= cyClient / cyChar ;
break;
//点一下方块下面:向下翻一页
case SB_PAGEDOWN:
iVscrollPos += cyClient / cyChar ;
break;
//拖动滚动条:wParam的高字节表明了滚动条的位置
case SB_THUMBPOSITION:
iVscrollPos = HIWORD (wParam) ;
break;
default:
break;
}
//这一句是为了
iVscrollPos = max (0, min (iVscrollPos, NUMLINES - 1)) ;
//如果滚动条位置改变
if (iVscrollPos != GetScrollPos (hwnd, SB_VERT))
{
//设置滚动条小方块的位置
SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;
//设为无效区,激励WM_PAINT消息
InvalidateRect (hwnd, NULL, TRUE) ;
}
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;
}
程序的思路是这样的:在创建窗口时,就要告诉系统你要加一个滚动条(垂直的),在WM_CREATE消息中,就要对滚动条进行一些基本的设置。WM_VSCROLL对滚动条做出相应,可以分为3部分:第一部分接收滚动条的消息;第二部分是让滚动条的那个小方块滚动到特定的位置,第三部分是页面的文字出现“滚动”的效果。
先看第一部分:滚动条的作用是什么:当点击一下向上或者向下的按钮时,会滚动一行文字;当点击滚动的小方块的上面或者下面时,会滚动一页文字;而拖着滚动条走,能滚动到指定的地方。到底是这几种操作的哪一种,是通过WM_VSCROLL消息的wParam的低字节传过来的。在程序中,使用iVscrollPos这个全局变量来记录滚动的位置,每一次对滚动条都会改变位置。
iVscrollPos = max (0, min (iVscrollPos, NUMLINES - 1)) ;是为了不让滚动条的位置超出一定为范围,大家可以把它屏蔽之后,将滚动条向上或者向下滚,就可以理解结果了。
而通过SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;就将滚动条的方块放到了可定位位置。而第三部分让页面滚动,则是WM_PAINT的作用了:其实当你看到第二行显示在原来第一行为位置上时,其实并不是系统没有输出第一行,而是第一行输出的位置已经到了客户区的上面,所以你看不到!第二行紧跟在第一行下面,刚好就放到了开始的位置。
这个滚动条有一个明显的缺点,就是他并不是我们希望看到的那样:滚动条的长度反应的当前页面占显示的内容总的内容的百分比,具体的说,就是:
页面方块的大小/滚动的长度=页面大小/整个范围=显示文件的数量/整个文件的数量
别看这只是一点小小的改动,但是他的实现方法却与上面的差别很大。总体上来说,他需要使用SCROLLINFO结构来记录滚动条的信息,然后使用GetScrollInfo获取信息,通过SetScrollInfo设置信息。在对滚动条的响应消息中,通过ScrollWindow来实现内容的滚动(而不像前面那个程序,在WM_PAINT消息下实现)。
我们在看看程序:
#include <windows.h>
#include "sysmets.h"
LRESULT CALLBACK WndProc (HWND,UINT,WPARAM,LPARAM);
int WINAPI WinMain(HINSTANCE hInstance, //当前实例句柄
HINSTANCE hPrevInstance, //先前实例句柄
LPSTR lpCmdLine, //命令行
int iCmdShow) //显示状态
{
static TCHAR szAppName[] = TEXT("显示系统内容");
//窗口句柄
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))
{
return -1;
}
//创建窗口
hwnd = CreateWindow(szAppName, //窗口类的名称,必须是已经注册的
TEXT("系统内容"), //窗口标题
WS_OVERLAPPEDWINDOW | WS_VSCROLL | WS_HSCROLL, //窗口风格,加入滚动条
CW_USEDEFAULT, //X坐标
CW_USEDEFAULT, //Y坐标
CW_USEDEFAULT, //宽度
CW_USEDEFAULT, //高度
NULL, //父窗口句柄
NULL, //菜单窗口句柄
hInstance, //高级版本的windos忽略
NULL);
//显示窗口
//ShowWindow(hwnd,SW_SHOWNA);
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)
{
//字符的宽度,大写字母宽度,字符高度
static int cxChar, cxCaps, cyChar ;
//窗口大小
static int cxClient, cyClient ;
//最大宽度
static int iMaxWidth;
//滚动条位置
static int iVertPos,iHorzPos,iPaintBeg,iPaintEnd;
HDC hdc;
//该变量用于索引sysmets.h中定义的结构体数组sysmetrics[]的每个元素
int i;
//输出文本的位置
int x,y;
//绘图结构
PAINTSTRUCT ps;
//这个结构包含了滚动条的信息
//通过SetScrollInfo函数设置信息,通过 GetScrollInfo 函数获取信息
SCROLLINFO si;
//字符串
TCHAR szBuffer [10];
//字体信息结构
TEXTMETRIC tm;
switch(message)
{
case WM_CREATE:
hdc = GetDC(hwnd);
//取得内定系统字体的文字大小,存在放在tm里
GetTextMetrics (hdc, &tm);
//平均字符宽
cxChar = tm.tmAveCharWidth ;
//大写字母的平均宽度
cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2 ;
//字符总高度:高度+行间距
cyChar = tm.tmHeight + tm.tmExternalLeading ;
ReleaseDC(hwnd,hdc);
//最大跨度 = 40个字符+22个大写字母
iMaxWidth = 40*cxChar+22*cxCaps;
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_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;
//max(0,当前位置+需要绘图的矩形区的最高点/字符的高度)
//iPaintBeg = max(0,iVertPos+ps.rcPaint.top/cyChar);
iPaintBeg = max(0,iVertPos);
//绘图结束的地方 = 当前位置+绘制去取的高度
iPaintEnd = min(NUMLINES -1,iVertPos+ps.rcPaint.bottom/cyChar);
for(i = iPaintBeg; i <= iPaintEnd;i++)
{
//绘图的x起始位置:1是自己设置的,设置越大离左边越宽
x = cxChar * (1 - iHorzPos) ;
//绘图的y起始位置。
y = cyChar * (i - iVertPos) ;
TextOut(hdc,x, y,sysmetrics[i].szLabel,lstrlen(sysmetrics[i].szLabel));
//从22个大写字母以后的位置输出,因为第一列最多只有20个大写字母
TextOut(hdc,x+22*cxCaps,y,sysmetrics[i].szDesc,lstrlen(sysmetrics[i].szDesc));
SetTextAlign(hdc,TA_RIGHT | TA_TOP);
TextOut(hdc,x+22*cxCaps+40*cxChar,y,szBuffer,wsprintf(szBuffer,TEXT("%5d"),GetSystemMetrics (sysmetrics[i].Index)));
SetTextAlign(hdc,TA_LEFT | TA_TOP);
}
EndPaint (hwnd, &ps) ;
return 0;
//垂直滚动条消息
case WM_VSCROLL:
//获取垂直滚动条信息
si.cbSize = sizeof(si);
//所有参数
si.fMask = SIF_ALL;
GetScrollInfo(hwnd,SB_VERT,&si);
//垂直位置
iVertPos = si.nPos ;
//通过滚动条消息的wParam表明滚动条的操作
switch(LOWORD (wParam))
{
case SB_TOP:
si.nPos = si.nMin;
break;
case SB_BOTTOM:
si.nPos = si.nMax;
break;
case SB_LINEUP:
si.nPos -= 1;
break;
case SB_LINEDOWN:
si.nPos +=1;
case SB_PAGEUP:
si.nPage -= 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);
//UpdateWindow (hwnd) ;
}
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;
}
程序中有几点需要注意:
1.程序中设置了垂直滚动条和水平滚动条。在WM_SIZE消息中,随着窗口的变化,滚动条的那个小方块的大小也在变化。
2.在每次使用GetScrollInfo或者SetScrollInfo之前,都必须有si.cbSize = sizeof(si);这是因为兼容的缘故。而且还得指明操作的是SCROLLINFO中的哪些内容,这通过fMask来控制。
3.程序中有几行代码很费解:
iPaintBeg = max(0,iVertPos+ps.rcPaint.top/cyChar);实际上,由于ps.rcPaint.top=0,这行代码也可以写为:iPaintBeg = max(0,iVertPos);
iPaintEnd = min(NUMLINES -1,iVertPos+ps.rcPaint.bottom/cyChar);画图的结束位置=滚动条当前的位置+绘制的行数,而行数=客户区宽度/每一行的宽度;
x = cxChar * (1 - iHorzPos) ;绘图的x坐标,那个1是为了不是太“顶格”,你也可以把它设置大一点,结果就很明显了。