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 |
TRUE或FALSE |
SCROLLINFO结构指针 |
本消息通过一个SCROLLINFO结构来同时指定控件的多种参数,具体指定哪些参数由结构中的fMask成员确定。wParam指定是否重绘控件,详见“SCROLLINFO结构” |
当用户在滚动条控件上进行各种操作时,其父窗口将收到WM_HSCROLL或WM_VSCROLL通知消息,同时wParam的低16位带有如下表的消息代码(nScrollCode),wParam的高16位带滚动框的指定位置(nPos),该值在消息代码等于SB_THUMBPOSITION或SB_THUMBTRACK时才有效。lParam带控件句柄(hwndScrollBar)。
应用程序可以根据消息代码做相应的操作,重新设置滚动框位置,控件本身是不会改变滚动框位置的。
消息代码 |
动作 |
响应 |
SB_LINEUP |
用户点击了向上(左)剪头 |
滚动框位置减一,客户窗口向上(左)滚动一行。 |
SB_LINEDOWN |
用户点击了向下(右)剪头 |
滚动框位置加一,客户窗口向下(右)滚动一行。 |
SB_PAGEUP |
用户点击了滚动框以上(左)剪杆 |
滚动框位置减去一个大单位,客户窗口向上(左)滚动一页。 |
SB_PAGEDOWN |
用户点击了滚动框以下(右)剪杆 |
滚动框位置加上一个大单位,客户窗口向下(右)滚动一页。 |
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 :nMin和nMax成员有效
nMin:滚动范围最小值
nMax:滚动范围最大值
nPage:页尺寸,用来确定比例滚动框的大小
nPos:滚动框的位置
nTrackPos:拖动时滚动框的位置,该参数只能查询,不能设置。
c#里的滚动条是封装得比较结实的,连滚动的消息都不提供出来,只能从wndproc里获取ox14,ox15的消息,在透明LISTVIEW那篇有提过