滚动条(2)

在用鼠标单击滚动条或者拖动卷动方块时,Windows给窗口消息处理程序发送WM_VSCROLL(供上下移动)和WM_HSCROLL(供左右移动)消息。在滚动条上的每个鼠标动作都至少产生两个消息,一条在按下鼠标按钮时产生,一条在释放按钮时产生。

和所有的消息一样,WM_VSCROLL和WM_HSCROLL也带有wParam和lParam消息参数。对于来自作为窗口的一部分而建立的滚动条消息,您可以忽略lParam;它只用于作为子窗口而建立的滚动条(通常在对话框内)。

wParam消息参数被分为一个低字组和一个高字组。wParam的低字组是一个数值,它指出了鼠标对滚动条进行的操作。这个数值被看作一个「通知码」。通知码的值由以SB(代表「scroll bar(滚动条)」)开头的标识符定义。以下是在WINUSER.H中定义的通知码:


  
  
  1. #define SB_LINEUP 0
  2. #define SB_LINELEFT 0
  3. #define SB_LINEDOWN 1
  4. #define SB_LINERIGHT 1
  5. #define SB_PAGEUP 2
  6. #define SB_PAGELEFT 2
  7. #define SB_PAGEDOWN 3
  8. #define SB_PAGERIGHT 3
  9. #define SB_THUMBPOSITION 4
  10. #define SB_THUMBTRACK 5
  11. #define SB_TOP 6
  12. #define SB_LEFT 6
  13. #define SB_BOTTOM 7
  14. #define SB_RIGHT 7
  15. #define SB_ENDSCROLL 8



       

包含LEFT和RIGHT的标识符用于水平滚动条,包含UP、DOWN、TOP和BOTTOM的标识符用于垂直滚动条。鼠标在滚动条的不同区域单击所产生的通知码如图4-7所示。

 

如果在滚动条的各个部位按住鼠标键,程序就能收到多个滚动条消息。当释放鼠标键后,程序会收到一个带有SB_ENDSCROLL通知码的消息。一般可以忽略这个消息,Windows不会去改变卷动方块的位置,而您可以在程序中呼叫SetScrollPos来改变卷动方块的位置。

当把鼠标的光标放在卷动方块上并按住鼠标键时,您就可以移动卷动方块。这样就产生了带有SB_THUMBTRACK和SB_THUMBPOSITION通知码的滚动条消息。在wParam的低字组是SB_THUMBTRACK时,wParam的高字组是使用者在拖动卷动方块时的目前位置。该位置位于卷动列范围的最小值和最大值之间。在wParam的低字组是SB_THUMBPOSITION时,wParam的高字组是使用者释放鼠标键后卷动方块的最终位置。对于其它的卷动列操作,wParam的高字组应该被忽略。

为了给使用者提供回馈,Windows在您用鼠标拖动卷动方块时移动它,同时您的程序会收到SB_THUMBTRACK消息。然而,如果不通过呼叫SetScrollPos来处理SB_THUMBTRACK或SB_THUMBPOSITION消息,在使用者释放鼠标键后,卷动方块会迅速跳回原来的位置。

程序能够处理SB_THUMBTRACK或SB_THUMBPOSITION消息,但一般不同时处理两者。如果处理SB_THUMBTRACK消息,在使用者拖动卷动方块时您需要移动显示区域的内容。而如果处理SB_THUMBPOSITION消息,则只需在使用者停止拖动卷动方块时移动显示区域的内容。处理SB_THUMBTRACK消息更好一些(但更困难),对于某些型态的数据,您的程序可能很难跟上产生的消息。

WINUSER.H表头文件还包括SB_TOP、SB_BOTTOM、SB_LEFT和SB_RIGHT通知码,指出滚动条已经被移到了它的最小或最大位置。然而,对于作为应用程序窗口一部分而建立的滚动条来说,永远不会接收到这些通知码。

在滚动条范围使用32位的值也是有效的,尽管这不常见。然而,wParam的高字组只有16位的大小,它不能适当地指出SB_THUMBTRACK和SB_THUMBPOSITION操作的位置。在这种情况下,需要使用GetScrollInfo函数(在下面描述)来得到信息。

在SYSMETS中加入卷动功能

前面的说明已经很详尽了,现在,要将那些东西动手做做看了。让我们开始时简单些,从垂直卷动着手,因为我们实在太需要垂直卷动了,而暂时还可以不用水平卷动。SYSMET2如程序4-3所示。这个程序可能是滚动条的最简单的应用。

程序4-3 SYSMETS2.C        


  
  
  1. /*------------------------------------------------------------------
  2. SYSMETS2.C -- System Metrics Display Program No. 2
  3. (c) Charles Petzold, 1998
  4. ------------------------------------------------------------------*/
  5. #include <windows.h>
  6. #include "sysmets.h"
  7. LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;
  8. int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
  9. PSTR szCmdLine, int iCmdShow)
  10. {
  11. static TCHAR szAppName[] = TEXT ( "SysMets2") ;
  12. HWND hwnd ;
  13. MSG msg ;
  14. WNDCLASS wndclass ;
  15. wndclass.style = CS_HREDRAW | CS_VREDRAW ;
  16. wndclass.lpfnWndProc = WndProc ;
  17. wndclass.cbClsExtra = 0 ;
  18. wndclass.cbWndExtra = 0 ;
  19. wndclass.hInstance = hInstance ;
  20. wndclass.hIcon = LoadIcon ( NULL, IDI_APPLICATION) ;
  21. wndclass.hCursor = LoadCursor ( NULL, IDC_ARROW) ;
  22. wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
  23. wndclass.lpszMenuName = NULL ;
  24. wndclass.lpszClassName = szAppName ;
  25. if (!RegisterClass (&wndclass))
  26. {
  27. MessageBox ( NULL, TEXT ( "This program requires Windows NT!"),
  28. szAppName, MB_ICONERROR) ;
  29. return 0 ;
  30. }
  31. hwnd = CreateWindow (szAppName, TEXT ( "Get System Metrics No. 2"),
  32. WS_OVERLAPPEDWINDOW | WS_VSCROLL,
  33. CW_USEDEFAULT, CW_USEDEFAULT,
  34. CW_USEDEFAULT, CW_USEDEFAULT,
  35. NULL, NULL, hInstance, NULL) ;
  36. ShowWindow (hwnd, iCmdShow) ;
  37. UpdateWindow (hwnd) ;
  38. while (GetMessage (&msg, NULL, 0, 0))
  39. {
  40. TranslateMessage (&msg) ;
  41. DispatchMessage (&msg) ;
  42. }
  43. return msg.wParam ;
  44. }
  45. LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
  46. {
  47. static int cxChar, cxCaps, cyChar, cyClient, iVscrollPos ;
  48. HDC hdc ;
  49. int i, y ;
  50. PAINTSTRUCT ps ;
  51. TCHAR szBuffer[ 10] ;
  52. TEXTMETRIC tm ;
  53. switch (message)
  54. {
  55. case WM_CREATE:
  56. hdc = GetDC (hwnd) ;
  57. GetTextMetrics (hdc, &tm) ;
  58. cxChar = tm.tmAveCharWidth ;
  59. cxCaps = (tm.tmPitchAndFamily & 1 ? 3 : 2) * cxChar / 2 ;
  60. cyChar = tm.tmHeight + tm.tmExternalLeading ;
  61. ReleaseDC (hwnd, hdc) ;
  62. SetScrollRange (hwnd, SB_VERT, 0, NUMLINES - 1, FALSE) ;
  63. SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;
  64. return 0 ;
  65. case WM_SIZE:
  66. cyClient = HIWORD (lParam) ;
  67. return 0 ;
  68. case WM_VSCROLL:
  69. switch (LOWORD (wParam))
  70. {
  71. case SB_LINEUP:
  72. iVscrollPos -= 1 ;
  73. break ;
  74. case SB_LINEDOWN:
  75. iVscrollPos += 1 ;
  76. break ;
  77. case SB_PAGEUP:
  78. iVscrollPos -= cyClient / cyChar ;
  79. break ;
  80. case SB_PAGEDOWN:
  81. iVscrollPos += cyClient / cyChar ;
  82. break ;
  83. case SB_THUMBPOSITION:
  84. iVscrollPos = HIWORD (wParam) ;
  85. break ;
  86. default :
  87. break ;
  88. }
  89. iVscrollPos = max ( 0, min (iVscrollPos, NUMLINES - 1)) ;
  90. if (iVscrollPos != GetScrollPos (hwnd, SB_VERT))
  91. {
  92. SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;
  93. InvalidateRect (hwnd, NULL, TRUE) ;
  94. }
  95. return 0 ;
  96. case WM_PAINT:
  97. hdc = BeginPaint (hwnd, &ps) ;
  98. for (i = 0 ; i < NUMLINES ; i++)
  99. {
  100. y = cyChar * (i - iVscrollPos) ;
  101. TextOut (hdc, 0, y,
  102. sysmetrics[i].szLabel,
  103. lstrlen (sysmetrics[i].szLabel)) ;
  104. TextOut (hdc, 22 * cxCaps, y,
  105. sysmetrics[i].szDesc,
  106. lstrlen (sysmetrics[i].szDesc)) ;
  107. SetTextAlign (hdc, TA_RIGHT | TA_TOP) ;
  108. TextOut (hdc, 22 * cxCaps + 40 * cxChar, y, szBuffer,
  109. wsprintf (szBuffer, TEXT ( "%5d"),
  110. GetSystemMetrics (sysmetrics[i].iIndex))) ;
  111. SetTextAlign (hdc, TA_LEFT | TA_TOP) ;
  112. }
  113. EndPaint (hwnd, &ps) ;
  114. return 0 ;
  115. case WM_DESTROY:
  116. PostQuitMessage ( 0) ;
  117. return 0 ;
  118. }
  119. return DefWindowProc (hwnd, message, wParam, lParam) ;
  120. }


 

新的CreateWindow呼叫在第三个参数中包含了WS_VSCROLL窗口样式,从而在窗口中加入了垂直滚动条,其窗口样式为:

WS_OVERLAPPEDWINDOW | WS_VSCROLL
       

WndProc窗口消息处理程序在处理WM_CREATE消息时增加了两条叙述,以设置垂直滚动条的范围和初始位置:

SetScrollRange (hwnd, SB_VERT, 0, NUMLINES - 1, FALSE) ;
       
SetScrollPos (hwnd, SB_VERT, iVscrollPos, TRUE) ;
       

sysmetrics结构具有NUMLINES行文字,所以滚动条范围被设定为0至NUMLINES-1。滚动条的每个位置对应于在显示区域顶部显示的一个文字行。如果卷动方块的位置为0,则第一行会被放置在显示区域的顶部。如果位置大于0,其它行就会出现在显示区域的顶部。当位置为NUMLINES-1时,则最后一行文字出现在显示区域的顶部。

为了有助于处理WM_VSCROLL消息,在窗口消息处理程序中定义了一个静态变量iVscrollPos,这一变量是滚动条内卷动方块的目前位置。对于SB_LINEUP和SB_LINEDOWN,只需要将卷动方块调整一个单位的位置。对于SB_PAGEUP和SB_PAGEDOWN,我们想移动一整面的内容,或者移动cyClient /cyChar个单位的位置。对于SB_THUMBPOSITION,新的卷动方块位置是wParam的高字组。SB_ENDSCROLL和SB_THUMBTRACK消息被忽略。

在程序依据收到的WM_VSCROLL消息计算出新的iVscrollPos值后,用min和max宏来调整iVscrollPos,以确保它在最大值与最小值之间。程序然后将iVscrollPos与呼叫GetScrollPos取得的先前位置相比较,如果卷动位置发生了变化,则使用SetScrollPos来进行更新,并且呼叫InvalidateRect使整个窗口无效。

InvalidateRect呼叫产生一个WM_PAINT消息。SYSMETS1在处理WM_PAINT消息时,每一行的y坐标计算公式为:

cyChar * i
       

在SYSMETS2中,计算公式为:

cyChar * (i - iVscrollPos)
       

循环仍然显示NUMLINES行文字,但是对于非零值的iVscrollPos是负数。程序实际上在显示区域以外显示这些文字行。当然,Windows不会显示这些行,因此屏幕显得干净和漂亮。

前面说过,我们一开始不想弄得太复杂,这样的程序代码很浪费,效率很低。下面我们对此加以修改,但是先要考虑在WM_VSCROLL消息之后更新显示区域的方法。

<br />本文来自【C语言中文网】:<a href="http://see.xidian.edu.cn/cpp/html/1112.html" target="_blank">http://see.xidian.edu.cn/cpp/html/1112.html</a>

猜你喜欢

转载自blog.csdn.net/qq_40627648/article/details/84844256