四、Windows子窗口控件的滚动条类别
我在第四章首次讨论了滚动条,也讨论了「窗口滚动条」和「滚动条控件」之间的一些区别。SYSMETS程序使用窗口滚动条,它出现在窗口的右边和底部。您可以在建立窗口时通过将标识符WS_VSCROLL、WS_HSCROLL或者两者都包含在窗口样式中,让窗口加上滚动条。现在我们准备建立一些滚动条控件,它们是能在父窗口的显示区域的任何地方出现的子窗口。您可以使用预先定义的窗口类别「scrollbar」以及两个滚动条样式SBS_VERT和SBS_HORZ中的一个来建立子窗口滚动条控件。
与按钮控件(以及将在后面讨论的编辑和清单方块控件)不同,滚动条控件不向父窗口发送WM_COMMAND消息,而是像窗口滚动条那样发送WM_VSCROLL和WM_HSCROLL消息。在处理卷动消息时,您可以通过lParam参数来区分窗口滚动条与滚动条控件。对子窗口滚动条其值为0,对于滚动条控件其值为滚动条窗口句柄。对窗口滚动条和滚动条控件来说,wParam参数的高字组和低字组的含义相同。
虽然窗口滚动条有固定的宽度,Windows使用CreateWindow呼叫中(或者在后面的MoveWindow呼叫中)给定的矩形尺寸来确定滚动条控件的尺寸。您可以建立细而长的滚动条控件,也可以建立短而粗的滚动条控件。
如果您想建立与窗口滚动条尺寸相同的滚动条控件,那么可以使用GetSystemMetrics取得水平滚动条的高度:
GetSystemMetrics (SM_CYHSCROLL) ;
        
或者垂直滚动条的宽度:
GetSystemMetrics (SM_CXVSCROLL) ;
        
根据Windows文件,滚动条窗样式标识符SBS_LEFTALIGN、SBS_RIGHTALIGN、SBS_TOP ALIGN和SBS_BOTTOMALIGN给出滚动条的标准尺寸,但是这些样式只在对话框中对滚动条有效。
对窗口滚动条,您可以使用同样的呼叫来建立滚动条控件的范围和位置:
SetScrollRange (hwndScroll, SB_CTL, iMin, iMax, bRedraw) ;
        
SetScrollPos (hwndScroll, SB_CTL, iPos, bRedraw) ;
        
SetScrollInfo (hwndScroll, SB_CTL, &si, bRedraw) ;
        
其区别在于:窗口滚动条将父窗口的句柄作为第一个参数,并且以SB_VERT或者SB_HORZ作为第二个参数。
令人吃惊的是,名为COLOR_SCROLLBAR的系统颜色不再用于滚动条。两端的按钮和小方块的颜色由COLOR_BTNFACE、COLOR_BTNHILIGHT、COLOR_BTNSHADOW、COLOR_BTNTEXT (用于小箭头)、COLOR_DKSHADOW和COLOR_BTNLIGHT决定。两端按钮之间区域的颜色由COLOR_BTNFACE和COLOR_BTNHIGHLIGHT决定。
如果您拦截了WM_CTLCOLORSCROLLBAR消息,那么可以在消息处理中传回画刷以取代该颜色。让我们来试一下。
COLORS1程序
为了解滚动条和静态子窗口的一些用法-也为了深入了解颜色-我们将使用COLORS1程序,如程序9-3所示。COLORS1在显示区域的左半部显示三种滚动条,并分别标以「Red」、「 Green」和「Blue」。当您挪动滚动条时,显示区域的右半部将变为三种原色混合而成的合成色,三种原色的数值显示在三个滚动条的下面。
        
COLORS1.C
        
/*------------------------------------------------------------------------
        
  COLORS1.C -- Colors Using Scroll Bars
        
                                  (c) Charles Petzold, 1998
        
-------------------------------------------------------------------------*/
        
#include <windows.h>
        
LRESULT CALLBACK WndProc           (HWND, UINT, WPARAM, LPARAM) ;
        
LRESULT CALLBACK ScrollProc(HWND, UINT, WPARAM, LPARAM) ;
        
int     idFocus ;
        
WNDPROC OldScroll[3] ;
        
int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
        
                                                         PSTR szCmdLine, int iCmdShow)
        
{
        
           static TCHAR szAppName[] = TEXT ("Colors1") ;
        
           HWND                                 hwnd ;
        
           MSG                                  msg ;
        
           WNDCLASS                      wndclass ;
        
   
        
          wndclass.style                               = CS_HREDRAW | CS_VREDRAW ;
        
           wndclass.lpfnWndProc                         = WndProc ;
        
           wndclass.cbClsExtra                          = 0 ;
        
           wndclass.cbWndExtra                          = 0 ;
        
           wndclass.hInstance                           = hInstance ;
        
           wndclass.hIcon                               = LoadIcon (NULL, IDI_APPLICATION) ;
        
           wndclass.hCursor                             = LoadCursor (NULL, IDC_ARROW) ;
        
           wndclass.hbrBackground               = CreateSolidBrush (0) ;
        
          wndclass.lpszMenuName                = NULL ;
        
           wndclass.lpszClassName               = szAppName ;
        
   
        
           if (!RegisterClass (&wndclass))
        
           {
        
                  MessageBox (  NULL, TEXT ("This program requires Windows NT!"),
        
                                                                        szAppName, MB_ICONERROR) ;
        
                  return 0 ;
        
           }
        
    hwnd = CreateWindow (szAppName, TEXT ("Color Scroll"),
        
                      WS_OVERLAPPEDWINDOW,
        
                      CW_USEDEFAULT, CW_USEDEFAULT,
        
                      CW_USEDEFAULT, CW_USEDEFAULT,
        
                     NULL, NULL, hInstance, NULL) ;
        
   
        
           ShowWindow (hwnd, iCmdShow) ;
        
           UpdateWindow (hwnd) ;
        
   
        
           while (GetMessage (&msg, NULL, 0, 0))
        
           {
        
                  TranslateMessage (&msg) ;
        
                  DispatchMessage  (&msg) ;
        
           }
        
          return msg.wParam ;
        
}
        
LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam,LPARAM lParam)
        
{
        
           static COLORREF crPrim[3] = { RGB (255, 0, 0), RGB (0, 255, 0),
        
              RGB (0, 0, 255) } ;
        
           static HBRUSH         hBrush[3], hBrushStatic ;
        
           static HWND           hwndScroll[3], hwndLabel[3], hwndValue[3], hwndRect ;
        
           static int                    color[3], cyChar ;
        
           static RECT           rcColor ;
        
           static TCHAR *szColorLabel[] = {    TEXT ("Red"), TEXT ("Green"),
        
       TEXT ("Blue") } ;
        
           HINSTANCE                     hInstance ;
        
           int                           i, cxClient, cyClient ;
        
           TCHAR                         szBuffer[10] ;
        
   
        
           switch (message)
        
           {
        
           case   WM_CREATE :
        
                  hInstance = (HINSTANCE) GetWindowLong (hwnd, GWL_HINSTANCE) ;
        
       
        
                          // Create the white-rectangle window against which the
        
                          // scroll bars will be positioned. The child window ID is 9.
        
        
        
                  hwndRect = CreateWindow (TEXT ("static"), NULL,
        
                       WS_CHILD | WS_VISIBLE | SS_WHITERECT,
        
                       0, 0, 0, 0,
        
                       hwnd, (HMENU) 9, hInstance, NULL) ;
        
        
        
                  for (i = 0 ; i < 3 ; i++)
        
                  {
        
                                  // The three scroll bars have IDs 0, 1, and 2, with
        
                                  // scroll bar ranges from 0 through 255.
        
             
        
                  hwndScroll[i] = CreateWindow (TEXT ("scrollbar"), NULL,
        
                               WS_CHILD | WS_VISIBLE |
        
                               WS_TABSTOP | SBS_VERT,
        
                                  0, 0, 0, 0,
        
               hwnd, (HMENU) i, hInstance, NULL) ;
        
             
        
                          SetScrollRange (hwndScroll[i], SB_CTL, 0, 255, FALSE) ;
        
                          SetScrollPos   (hwndScroll[i], SB_CTL, 0, FALSE) ;
        
             
        
      // The three color-name labels have IDs 3, 4, and 5,
        
     // and text strings "Red", "Green", and "Blue".
        
             
        
           hwndLabel [i] = CreateWindow (TEXT ("static"), zColorLabel[i],
        
                                   WS_CHILD | WS_VISIBLE | SS_CENTER,
        
                                     0, 0, 0, 0,
        
                                     hwnd, (HMENU) (i + 3),
        
                                     hInstance, NULL) ;
        
             
        
                          // The three color-value text fields have IDs 6, 7,
        
                          // and 8, and initial text strings of "0".
        
             
        
hwndValue [i] = CreateWindow (TEXT ("static"), TEXT ("0"),
        
                                WS_CHILD | WS_VISIBLE | SS_CENTER,
        
                                0, 0, 0, 0,
        
                               hwnd, (HMENU) (i + 6),
        
                                hInstance, NULL) ;
        
             
        
                          OldScroll[i] = (WNDPROC) SetWindowLong (hwndScroll[i],
        
                                 GWL_WNDPROC, (LONG) ScrollProc) ;
        
             
        
                          hBrush[i] = CreateSolidBrush (crPrim[i]) ;
        
                  }
        
        
        
                    hBrushStatic = CreateSolidBrush (
        
                                GetSysColor (COLOR_BTNHIGHLIGHT)) ;
        
        
        
            cyChar = HIWORD (GetDialogBaseUnits ()) ;
        
           return 0 ;
        
        
        
           case   WM_SIZE :
        
                  cxClient = LOWORD (lParam) ;
        
                  cyClient = HIWORD (lParam) ;
        
                  SetRect (&rcColor, cxClient / 2, 0, cxClient, cyClient) ;
        
        
        
                  MoveWindow (hwndRect, 0, 0, cxClient / 2, cyClient, TRUE) ;
        
        
        
                  for (i = 0 ; i < 3 ; i++)
        
           {
        
                         MoveWindow (hwndScroll[i],
        
                                 (2 * i + 1) * cxClient / 14, 2 * cyChar,
        
                                  cxClient / 14, cyClient - 4 * cyChar, TRUE) ;
        
             
        
                          MoveWindow (hwndLabel[i],
        
                                  (4 * i + 1) * cxClient / 28, cyChar / 2,
        
                                 cxClient / 7, cyChar, TRUE)
        
             
        
                          MoveWindow (hwndValue[i],
        
                                  (4 * i + 1) * cxClient / 28,
        
                                  cyClient - 3 * cyChar / 2,
        
                                  cxClient / 7, cyChar, TRUE) ;
        
           }
        
                  SetFocus (hwnd) ;
        
                  return 0 ;
        
        
        
           case   WM_SETFOCUS :
        
                  SetFocus (hwndScroll[idFocus]) ;
        
                  return 0 ;
        
        
        
           case   WM_VSCROLL :
        
                 i = GetWindowLong ((HWND) lParam, GWL_ID) ;
        
        
        
                  switch (LOWORD (wParam))
        
                  {
        
                  case   SB_PAGEDOWN :
        
                                         color[i] += 15 ;
        
                           // fall through
        
                  case   SB_LINEDOWN :
        
                                       color[i] = min (255, color[i] + 1) ;
        
                                         break ;
        
             
        
                  case   SB_PAGEUP :
        
                                         color[i] -= 15 ;
        
                           // fall through
        
                  case   SB_LINEUP :
        
                                         color[i] = max (0, color[i] - 1) ;
        
                                         break ;
        
             
        
                  case   SB_TOP :
        
                                         color[i] = 0 ;
        
                                         break ;
        
             
        
                  case   SB_BOTTOM :
        
                                         color[i] = 255 ;
        
                                         break ;
        
             
        
                  case   SB_THUMBPOSITION :
        
                  case   SB_THUMBTRACK :
        
                                         color[i] = HIWORD (wParam) ;
        
                                         break ;
        
             
        
                  default :
        
                                         break ;
        
                  }
        
                 SetScrollPos  (hwndScroll[i], SB_CTL, color[i], TRUE) ;
        
                  wsprintf (szBuffer, TEXT ("%i"), color[i]) ;
        
                  SetWindowText (hwndValue[i], szBuffer) ;
        
        
        
                  DeleteObject ((HBRUSH)
        
                                  SetClassLong (hwnd, GCL_HBRBACKGROUND, (LONG)
        
                  CreateSolidBrush (RGB (color[0], color[1], color[2])))) ;
        
        
        
                  InvalidateRect (hwnd, &rcColor, TRUE) ;
        
                 return 0 ;
        
        
        
           case   WM_CTLCOLORSCROLLBAR :
        
                  i = GetWindowLong ((HWND) lParam, GWL_ID) ;
        
                  return (LRESULT) hBrush[i] ;
        
             
        
           case   WM_CTLCOLORSTATIC :
        
                  i = GetWindowLong ((HWND) lParam, GWL_ID) ;
        
             
        
    if (i >= 3 && i <= 8)                       // static text controls
        
           {
        
                          SetTextColor ((HDC) wParam, crPrim[i % 3]) ;
        
                          SetBkColor ((HDC) wParam, GetSysColor (COLOR_BTNHIGHLIGHT));
        
                          return (LRESULT) hBrushStatic ;
        
           }
        
           break ;
        
           case   WM_SYSCOLORCHANGE :
        
                 DeleteObject (hBrushStatic) ;
        
                  hBrushStatic = CreateSolidBrush (GetSysColor(COLOR_BTNHIGHLIGHT)) ;
        
                  return 0 ;
        
           case   WM_DESTROY :
        
                  DeleteObject ((HBRUSH)
        
                                  SetClassLong (hwnd, GCL_HBRBACKGROUND, (LONG)
        
                                         GetStockObject (WHITE_BRUSH))) ;
        
             
        
                  for (i = 0 ; i < 3 ; i++)
        
                                         DeleteObject (hBrush[i]) ;
        
             
        
                  DeleteObject (hBrushStatic) ;
        
                  PostQuitMessage (0) ;
        
                  return 0 ;
        
    }
        
           return DefWindowProc (hwnd, message, wParam, lParam) ;
        
}
        
   
        
LRESULT CALLBACK ScrollProc (HWND hwnd, UINT message,
        
                          WPARAM wParam, LPARAM lParam)
        
{
        
           int id = GetWindowLong (hwnd, GWL_ID) ;
        
           switch (message)
        
           {
        
           case WM_KEYDOWN :
        
                  if (wParam == VK_TAB)
        
                                         SetFocus (GetDlgItem (GetParent (hwnd),
        
                   (id + (GetKeyState (VK_SHIFT) < 0 ? 2 : 1)) % 3)) ;
        
                  break ;
        
           case   WM_SETFOCUS :
        
                 idFocus = id ;
        
                  break ;
        
           }
        
           return CallWindowProc (OldScroll[id], hwnd, message, wParam,lParam) ;
        
}
        
COLORS1利用子窗口进行工作,该程序使用10个子窗口控件:3个滚动条、6个静态文字窗口和1个静态矩形框。COLORS1拦截WM_CTLCOLORSCROLLBAR消息来给红、绿、蓝3个滚动条的内部着色,并拦截WM_CTLCOLORSTATIC消息来着色静态文字。
您可以使用鼠标或者键盘来挪动滚动条,从而利用COLORS1作为一种实验颜色显示的开发工具,为您自己的Windows程序选择漂亮的颜色(或者,您可能更喜欢难看的颜色)。COLORS1的显示如图9-3所示。不幸的是,这些颜色在印表纸上被显示为不同深浅的灰色。
	
COLORS1不处理WM_PAINT消息,所有的工作几乎都是由子窗口完成的。
显示区域右半部显示的颜色实际上是窗口的背景颜色。SS_WHITERECT样式的静态子窗口显示在显示区域的左半部。三个滚动条是SBS_VERT样式的子窗口控件,它们被定位在SS_WHITERECT子窗口的顶部。另外六个SS_CENTER样式(居中文字)的静态子窗口提供卷标和颜色值。COLORS1在WinMain函数中用CreateWindow建立它的普通重迭式窗口和10个子窗口。SS_WHITERECT和SS_CENTER静态窗口使用窗口类别「static」;三个滚动条使用窗口类别「scrollbar」。
CreateWindow呼叫中的x位置、y位置、宽度和高度参数最初设为0,因为位置和大小都取决于显示区域的尺寸,而它目前尚未确定。COLORS1的窗口消息处理程序在接收到WM_SIZE消息时,就使用MoveWindow给10个子窗口重新确定大小。所以,每当您对COLORS1窗口进行缩放时,滚动条的尺寸就会按比例变化。
当WndProc窗口消息处理程序收到WM_VSCROLL消息时,lParam参数的高字组就是子窗口的句柄。我们可以使用GetWindowWord来得到子窗口的ID:
i = GetWindowLong ((HWND) lParam, GWL_ID) ;
        
对于这三个滚动条,我们已经按习惯将其ID设为0、1、2,所以WndProc能区别出是哪个滚动条在产生消息。
由于子窗口的句柄在建立时就被储存在数组中,所以WndProc就能对相对应的滚动条消息进行处理,并通过呼叫SetScrollPos来设定相对应的新值:
SetScrollPos (hwndScroll[i], SB_CTL, color[i], TRUE) ;
        
WndProc也改变滚动条底部子窗口的文字:
wsprintf (szBuffer, TEXT ("%i"), color[I]) ;
        
SetWindowText (hwndValue[i], szBuffer) ;
