windows滚动条

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)。






你可能感兴趣的:(windows滚动条)