1:滚动条处理方法分类
windows中使用滚动条的函数有两类,一种比较古老的,但也是有效的,其调用函数如下:
bool SetScrollRange(hwnd,iBar,iMin,iMax,bRedraw);
int SetScrollPos(hwnd,iBar,iPos,bRedraw);
bool GetScrollRange(hwnd,ibar,lpiMin,lpiMax);
int GetScrollPos(hwnd,ibar);
另外一种是比较新颖,功能强大并且函数较少的,其函数如下:
int SetScrollInfo(hwnd,ibar,*si,bRedraw);
bool GetScrollInfo(hwnd,ibar,*si);
第三个参数是一种SCROLLINFO的结构体
typedef struct tagSCROLLINFO {
UINT cbSize;
UINT fMask;
int nMin;
int nMax;
UINT nPage;
int nPos;
int nTrackPos;
} SCROLLINFO, *LPSCROLLINFO; typedef SCROLLINFO CONST *LPCSCROLLINFO
fMask的取值:
The fMask member can be one or more of the following values.
SIF_DISABLENOSCROLL:
Disables the scroll bar instead of removing it, if the scroll bar's new parameters make the scroll bar unnecessary.
SIF_PAGE:
Sets the scroll page to the value specified in the nPage member of the SCROLLINFO structure pointed to by lpsi.
SIF_POS:
Sets the scroll position to the value specified in the nPos member of the SCROLLINFO structure pointed to by lpsi.
SIF_RANGE:
Sets the scroll range to the value specified in the nMin and nMax members of the SCROLLINFO structure pointed to by lpsi.
2:与滚动条相关的消息
#define SB_LINEUP 0
#define SB_LINELEFT 0
#define SB_LINEDOWN 1
#define SB_LINERIGHT 1
#define SB_PAGEUP 2
#define SB_PAGELEFT 2
#define SB_PAGEDOWN 3
#define SB_PAGERIGHT 3
#define SB_THUMBPOSITION 4
#define SB_THUMBTRACK 5
#define SB_TOP 6
#define SB_LEFT 6
#define SB_BOTTOM 7
#define SB_RIGHT 7
#define SB_ENDSCROLL 8
3:消息处理范围
滚动条(系统做的事)
(1)处理所有滚动条鼠标事件
(2)当使用者在滚动条内单击鼠标时,提供一种「反相显示」的闪烁
(3) 当使用者在滚动条内拖动卷动方块时,移动卷动方块
(4)为包含滚动条窗口的窗口消息处理程序发送滚动条消息
滚动条(自己做的事)
(1)初始化滚动条的范围和位置
(2) 处理窗口消息处理程序的滚动条消息
(3) 更新滚动条内卷动方块的位置
(4) 更改显示区域的内容以响应对滚动条的更改
4:滚动时显示的方法:
对于比较古老的滚动条显示采用
if (iVscrollPos != GetScrollPos (hwnd, SB_VERT))
{
SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;
InvalidateRect (hwnd, NULL, TRUE) ;
}
判断当前位置和滚动条先前位置是否相等,如果不相等,则设置滚动条位置,并且设置整个客户区无效,此时会发送WM_PAINT消息,在WM_PAINT中,利用
for (i = 0 ; i < NUMLINES ; i++)
{
y = cyChar * (i - iVscrollPos) ;
TextOut (hdc, 0, y,sysmetrics[i].szLabel,lstrlen (sysmetrics[i].szLabel)) ;
……}来显示文字,当y<0, 文字不会显示,所以只会显示滚动条开始位置的信息。
对于较新的滚动条函数采用
if (si.nPos != iHorzPos)
{
ScrollWindow (hwnd, cxChar * (iHorzPos - si.nPos), 0, NULL, NULL) ;
}//没用设置无效区。
WM_PAINT:
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) ;
TextOut (……);
……};
ScrollWindow(hwnd,x,y,lpRect,lpClipRect);
hWnd
[in]客户区域将被滚动的窗口的 句柄。
XAmount
[in]指定水平滚动的距离,以设备单位计。如果 窗口类风格为CS_OWNDC或CS_CLASSDC,则此参数则使用逻辑单位而非设备单位。当向左滚动 窗体内容时,参数值必须为负。
YAmount
[in]指定垂直滚动的距离,以设备单位计。如果窗口类风格为CS_OWNDC或CS_CLASSDC,则此参数则使用逻辑单位而非设备单位。当向上滚动 窗体内容时,参数值必须为负。
lpRect
[in]指向 RECT结构的 指针,该结构指定了将要滚动的客户区范围。若此参数为NULL,则整个客户区域将被滚动。
lpClipRect
[in]指向 RECT结构的指针,该结构指定了要滚动的 裁剪区域。只有这个矩形中的位才会被滚动。在矩形之外的位不会被影响,即使它们是在lpRect矩形之内。(见代码"测试一")假如lpClipRect为NULL,则不会在滚动矩形上进行裁剪。
5.完整的一个流程
1:要在窗口中使用滚动条,首先要在CreateWindow中的第3个参数中设置窗口显示的类型
WS_VSCROLL:显示纵向的滚动条
WS_HSCROLL:显示纵向的滚动条
或者用“|”将其类型进行逻辑或运算。
2:要获取字体的相关信息:
在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);
//SetScrollRange(hwnd,SB_VERT,0,NUMLINES-1,FALSE);
//SetScrollPos(hwnd,SB_VERT,iVscrollPos,TRUE);
iMaxWidth=40*cxChar+22*cxCaps;
通过以上的语句可以得到系统字体的宽度,高度,以及客户区的宽度。
3:获取滚动条的取值范围,每页的行数。这一切主要是在SCROLLINFO的结构体中完成。
是在WM_SIZE的消息响应中完成
代码如下:
cxClient=LOWORD(lParam);
cyClient=HIWORD(lParam);
//Set vertical scroll bar range and page size
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);
4:获取滚动条的位置信息;
这个要WM_SCROLL中完成。
si.cbSize=sizeof(si);
si.fMask=SIF_ALL;
GetScrollInfo(hwnd,SB_VERT,&si);
iVertPos=si.nPos;
switch(LOWORD(wParam))
{
case SB_TOP:
si.nPos=si.nMin;
break;
case SB_BOTTOM:
si.nPos=si.nMax;
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;
}
//iVscrollPos=max(0,min(iVscrollPos,NUMLINES-1));
si.fMask=SIF_POS;
SetScrollInfo(hwnd,SB_VERT,&si,TRUE);
GetScrollInfo(hwnd,SB_VERT,&si);
if(si.nPos!=iVertPos)
{
//SetScrollPos(hwnd,SB_VERT,iVscrollPos,TRUE);
//InvalidateRect(hwnd,NULL,TRUE);
ScrollWindow(hwnd,0,cyChar*(iVertPos-si.nPos),NULL,NULL);
UpdateWindow(hwnd);
}
5滚动条以及显示文本的绘制:
如果将滚动条的范围限定在为等同文本的行数,则当滚动条的最大值等同文本行时,其只显示该文本的最后一行,这看起来就不爽,解决方案如下:
iPaintBeg=max(0,iVertPos+ps.rcPaint.top/cyChar);
iPaintEnd=min(NUMLINES-1,iVertPos+ps.rcPaint.bottom/cyChar);
这两句话的含义就是将文本显示的范围定为:iVertPos-》iVertPos+(页面可显示的行数)
因为要重新绘制的无效区域就是整个客户区,所以rc.Paint.top=0;rc.PaintBottom=si.nPage.
当文本的行数少于页面的行数时,显示的行数范围就是(0-》NUMLINES-1)。