WinForm:关于滚动条的基础知识

以下是windows 程序設计里一段代码,简单的说明设置SCORLL,主要是几个API,二百多行代码,抄写了好久好久,不忍没掉,记录下来。。。。

 C#对滚动条是做了好多工作,我们无论在那个控件里都可以简单通过属性调用,但如果该控件到了自定义时,就比较麻烦了

 而在此也记录点滚动条的基础知识,总结备用吧。。。。。。

 

LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM LParam)

{

     static int cxChar,cxCaps,cyChar,cxClient,cyClient,iMaxWidth;

     int i,x,y,iVertPos,iHorzPos,iPaintBeg,iPaintEnd;

     PAINTSTRUCT ps;

     TCHAR szBuffer[10];

     TEXTMETRIC tm;

     HDC hdc;

     //PAINTSTRUCT ps;

     RECT rect;

     SCROLLINFO si;

 

 

     switch(message)

     {

     case WM_CREATE:

         hdc=GetDC(hwnd);

         GetTextMetrics(hdc,&tm);                       //调用系统,获取默认字体

         cxChar=tm.tmAveCharWidth;

         cyChar=tm.tmHeight+tm.tmExternalLeading;       //将高度包括间距,还有宽度保存起来

         ReleaseDC(hwnd,hdc);

         return 0;

 

     case WM_SIZE:

         cxClient=LOWORD(LParam);       //低位,

         cyClient=HIWORD(LParam);       //高位

                                        //此时的lparam的高字节保存高度,低字节保存宽度。

 

         si.cbSize=sizeof(si);

         si.fMask=SIF_RANGE|SIF_PAGE;   //SIF_RANGE必须把SCORLL(nMin,nMax)设置为所需的范围

         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: /* WM_HSCROLL水平滚动消息的值是&H114;WM_VSCROLL垂直滚动消息的值是&H115*/

         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;

              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:   /*B_THUMBTRACK 用户正在拖动滚动框 设定滚动框到指定位置。

                                  客户窗口滚动到指定位置。如果应用程序需要快速浏览窗口,

                                可以响应本消息重绘窗口,如果不需要快速浏览,可以等待收到SB_THUMBPOSITION消息时重绘窗口。*/

              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:     //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;

 

         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(hdc,x,y,sysmetrics[i].szLabel,lstrlen(sysmetrics[i].szLabel));

 

              SetTextAlign(hdc,TA_RIGHT|TA_TOP);

 

              TextOut(hdc,x+22*cxCaps+60*cxChar,y,szBuffer,wsprintf(szBuffer,TEXT("%5d"),GetSystemMetrics(sysmetrics[i].iIndex)));

 

              SetTextAlign(hdc,TA_LEFT|TA_TOP);

         }

 

     /*   GetClientRect(hwnd,&rect);       DrawText(hdc,TEXT("Hello ,windows 98"),-1,&rect,DT_SINGLELINE|DT_CENTER|DT_VCENTER);*/

         EndPaint(hwnd,&ps);

         return 0;

     case WM_DESTROY:

             

         PostQuitMessage(0);

         return 0;

     }

     return DefWindowProc(hwnd,message,wParam,LParam);

}

 

 

应用程序可以通过调用SendMessage向控件发送如下消息来设定控件各种参数。

uMsg

wParam

lParam

说明

SBM_ENABLE_ARROWS

ESB_DISABLE_BOTH

0

禁止双向滚动剪头

ESB_DISABLE_DOWN

0

禁止向下滚动剪头

ESB_DISABLE_LTUP

0

禁止向上和向左滚动剪头

ESB_DISABLE_LEFT

0

禁止向左滚动剪头

ESB_DISABLE_RTDN

0

禁止向下和向右滚动剪头

ESB_DISABLE_UP

0

禁止向上滚动剪头

ESB_ENABLE_BOTH

0

允许双向滚动剪头(撤消各种禁止)

SBM_SETPOS

指定位置

TRUE

设置滚动框位置,并重绘控件

FALSE

设置滚动框位置,不重绘控件

SBM_SETRANGE

最小值

最大值

设置滚动框位置的变化范围

SBM_SETRANGEREDRAW

最小值

最大值

设置滚动框位置的变化范围,并重绘控件

SBM_SETSCROLLINFO

TRUEFALSE

SCROLLINFO结构指针

本消息通过一个SCROLLINFO结构来同时指定控件的多种参数,具体指定哪些参数由结构中的fMask成员确定。wParam指定是否重绘控件,详见“SCROLLINFO结构

  当用户在滚动条控件上进行各种操作时,其父窗口将收到WM_HSCROLLWM_VSCROLL通知消息,同时wParam的低16位带有如下表的消息代码(nScrollCode)wParam的高16位带滚动框的指定位置(nPos),该值在消息代码等于SB_THUMBPOSITIONSB_THUMBTRACK时才有效。lParam带控件句柄(hwndScrollBar)
  应用程序可以根据消息代码做相应的操作,重新设置滚动框位置,控件本身是不会改变滚动框位置的。

消息代码

动作

响应

SB_LINEUP
SB_LINELEFT

用户点击了向上()剪头

滚动框位置减一,客户窗口向上()滚动一行。
注:这两个代码数值相等,因此可以混用,下同。

SB_LINEDOWN
SB_LINERIGHT

用户点击了向下()剪头

滚动框位置加一,客户窗口向下()滚动一行。

SB_PAGEUP
SB_PAGELEFT

用户点击了滚动框以上()剪杆

滚动框位置减去一个大单位,客户窗口向上()滚动一页。

SB_PAGEDOWN
SB_PAGERIGHT

用户点击了滚动框以下()剪杆

滚动框位置加上一个大单位,客户窗口向下()滚动一页。

SB_THUMBPOSITION

用户拖动并释放滚动框到指定位置

设定滚动框到指定位置。客户窗口滚动到指定位置。

SB_THUMBTRACK

用户正在拖动滚动框

设定滚动框到指定位置。客户窗口滚动到指定位置。如果应用程序需要快速浏览窗口,可以响应本消息重绘窗口,如果不需要快速浏览,可以等待收到SB_THUMBPOSITION消息时重绘窗口。

SB_ENDSCROLL

用户释放按下剪头或剪杆的鼠标

无须做任何响应

  应用程序可以通过调用SendMessage向控件发送如下消息来取得当前控件各种参数。

uMsg

wParam

lParam

说明

SBM_GETPOS

0

0

返回滚动框当前位置。

SBM_GETRANGE

最小值地址指针

最大值地址指针

在指定地址中填入32位的滚动框位置的变化范围

SBM_GETSCROLLINFO

0

SCROLLINFO结构指针

在一个SCROLLINFO结构中返回控件的多种参数,必须事先设定结构的fMask成员来确定具体要取得哪些参数。详见“SCROLLINFO结构

  当控件需要重画时向每父窗口发送WM_CTLCOLORSCROLLBAR消息,同时在wParam中带控件的设备场景句柄(hDC)lParam中带控件句柄。如果应用程序响应这个消息并返回一个画刷(brush)句柄,控件将根据这个句柄绘制背景色。

SCROLLINFO结构:

SCROLLINFO STRUCT

 cbSize        DWORD     

 fMask         DWORD     

 nMin          DWORD     

 nMax          DWORD     

 nPage         DWORD     

 nPos          DWORD     

 nTrackPos     DWORD     

SCROLLINFO ENDS

成员说明:
  cbSize: SCROLLINFO结构长度字节数,该值在设置和查询参数时都必须填写。
  fMask: 指定结构中的哪些成员是有效,该值共有如下5种选择,可以选择多种用“OR”组合起来,该值在设置和查询参数时都必须填写。
    SIF_ALL      :整个结构都有效
    SIF_DISABLENOSCROLL:该值仅在设定参数时使用,视控件参数设定的需要来对本结构的成员进行取舍。
    SIF_PAGE      :nPage成员有效
    SIF_POS      :nPos成员有效
    SIF_RANGE     :nMinnMax成员有效
  nMin:滚动范围最小值
  nMax:滚动范围最大值
  nPage:页尺寸,用来确定比例滚动框的大小
  nPos:滚动框的位置
  nTrackPos:拖动时滚动框的位置,该参数只能查询,不能设置。

 

 c#里的滚动条是封装得比较结实的,连滚动的消息都不提供出来,只能从wndproc里获取ox14,ox15的消息,在透明LISTVIEW那篇有提过

 

posted @ 2008-09-14 22:43  yellowyu  阅读(2740)  评论(1编辑  收藏  举报