1.根据窗体的大小和内容区大小决定是否需要展示滚动条
若rtWindow.width > rtContent.width,需要创建水平滚动条;若rtWindow.height > rtContent.height,需要创建垂直滚动条。
2.滚动条信息设置
SetScrollInfo可以设置滚动条的信息,重点对
SCROLLINFO结构进行设置,其中nMin=0即可,nMax最好设置成客户区的大小,nPage最好设置成窗体大小,事实上windows就是根据nPage/(nMax-nMin)来确定滚动条滑块的大小。
3.消息回调函数设计
因为WM_HSCROLL和WM_VSCROLL道理一致,因此只介绍WM_HSCROLL的设计,直接看下面的代码即可。
VOID CShowResWnd::OnHScrollBar(WPARAM wParam, LPARAM lParam)
{
SCROLLINFO si;
si.cbSize = sizeof(SCROLLINFO);
si.fMask = SIF_ALL;
::GetScrollInfo(hWnd, SB_HORZ, &si);
RECT rt;
::GetWindowRect(hShowDlg, &rt);
POINT pt;
pt.x = rt.left, pt.y = rt.top;
::ScreenToClient(hWnd, &pt);
int nSBCode = (int)LOWORD(wParam);
short int nPos = (short int)HIWORD(wParam);
switch (nSBCode)
{
case SB_LINELEFT:
si.nPos -= si.nPage/10;
if (si.nPos< si.nMin)
si.nPos = si.nMin;
break;
case SB_LINERIGHT:
si.nPos += si.nPage/10;
if (si.nPos> (si.nMax+1-si.nPage))
si.nPos = si.nMax+1-si.nPage;
break;
case SB_PAGELEFT:
si.nPos -= si.nPage;
if (si.nPos< si.nMin)
si.nPos = si.nMin;
break;
case SB_PAGERIGHT:
si.nPos += si.nPage;
if (si.nPos> (si.nMax+1-si.nPage))
si.nPos = si.nMax+1-si.nPage;
break;
case SB_THUMBTRACK:
si.nPos = nPos;
break;
case SB_THUMBPOSITION:
si.nPos = nPos;
break;
case SB_LEFT:
si.nPos = si.nMin;
break;
case SB_RIGHT:
si.nPos = si.nMax+1-si.nPage;
break;
case SB_ENDSCROLL:
return;
}
SetScrollInfo(hWnd, SB_HORZ, &si, TRUE);
::SetWindowPos(hShowDlg, 0, -si.nPos, pt.y, 0, 0, SWP_SHOWWINDOW | SWP_NOZORDER | SWP_NOSIZE);
return;
}
代码中的hShowDlg和hWnd分别对应上文提到的内容区和窗体,尤其需要注意,滚动条的位置即si.nPos不可能为nMax,最多是nMax+1-nPage,如果是nMax-nPage还好理解,但是在调试过程中发现需要+1才能设计成我们习惯使用的滚动条,这肯定是微软公司设计的缺陷。
si.nPos是上次滑块的位置,不是本次的位置。nPos只能由SB_THUMBTRACK和SB_THUMBPOSITION来用,其它都是0,没有意义。
至此,就可以设计成滚动条动态出现,滑块大小动态改变,移动滑块可以展示指定位置的区域,点击滚动条空白处可以翻页,点击滚动条的两端内容移动整页的1/10,并且滑块到达滚动条的两端就不再移动。