在CreateWindow函數中指定窗口類別爲「static」,您就可以建立靜態文字的子窗口控件。這些子窗口非常「文靜」。它既不接收鼠標或鍵盤輸入,也不向父窗口發送WM_COMMAND消息。
當您在靜態子窗口上移動或者按下鼠標時,這個子窗口將攔截WM_NCHITTEST消息並將HTTRANSPARENT的值傳回給Windows,這將使Windows向其下層窗口,通常是它的父窗口,發送相同的WM_NCHITTEST消息。父窗口常常將該消息傳遞給DefWindowProc,在這裏,它被轉換爲顯示區域的鼠標消息。
前六個靜態窗口樣式只簡單地在子窗口的顯示區域內畫一個矩形或者邊框。在下表的上部,「RECT」靜態樣式(左列)是填入圖樣的矩形樣式;三個「FRAME」樣式(右列)是沒有填入圖樣的矩形輪廓:
SS_BLACKRECT SS_GRAYRECT SS_WHITERECT |
SS_BLACKFRAME SS_GRAYFRAME SS_WHITEFRAME |
「BLACK」、「GRAY」、「WHITE」並不意味着黑、灰和白色,這些顏色是由系統顏色決定的,如表9-4所示。
表9-4 |
靜態控件 |
系統顏色 |
BLACK |
COLOR_3DDKSHADOW |
GRAY |
COLOR_BTNSHADOW |
WHITE |
COLOR_BTNHIGHLIGHT |
對這些樣式,CreateWindow呼叫中的窗口文字字段被忽略。矩形的左上角開始於x位置座標和y位置座標,這些座標都相對於父窗口。您也可以使用SS_ETCHEDHORZ、SS_ETCHEDVERT或者SS_ETCHEDFRAME ,採用灰色和白色建立一個形似陰影的邊框。
靜態類別也包括了三種文字樣式:SS_LEFT、SS_RIGHT和SS_CENTER。它們建立左對齊、置右對齊和居中文字。文字在CreateWindow呼叫的窗口文字參數中給出,並且在以後可以用SetWindowText來改變它。當靜態控件的窗口消息處理程序顯示文字時,它使用DrawText函數以及DT_WORDBREAK、DT_NOCLIP和DT_EXPANDTABS參數。文字在子窗口的矩形內可以按文字進行換行。
這三種文字樣式子窗口的背景通常爲COLOR_BTNFACE,而文字本身是COLOR_WINDOWTEXT。在攔截WM_CTLCOLORSTATIC消息時,您可以通過呼叫SetTextColor來改變文字顏色,通過SetBkColor來改變背景顏色,並傳回背景畫刷句柄。後面的COLORS1程序展示了這一點。
最後,靜態類別還包括了窗口樣式SS_ICON和SS_USERITEM,但是當它們被用作子窗口控件時卻沒有任何意義。我們在討論對話框時還要提及它們。
我在第四章首次討論了滾動條,也討論了「窗口滾動條」和「滾動條控件」之間的一些區別。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所示。不幸的是,這些顏色在印表紙上被顯示爲不同深淺的灰色。
圖9-3 COLORS1的屏幕顯示 |
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) ;
自動鍵盤接口
滾動條控件也能處理鍵盤輸入,但是只有在擁有輸入焦點時才行。下表說明怎樣將鍵盤光標鍵轉變爲捲動消息:
表9-5 |
遊標鍵 |
捲動消息的wParam值 |
Home |
SB_TOP |
End |
SB_BOTTOM |
Page Up |
SB_PAGEUP |
Page Down |
SB_PAGEDOWN |
左或上 |
SB_LINEUP |
右或下 |
SB_LINEDOWN |
事實上,SB_TOP和SB_BOTTOM捲動消息只能用鍵盤產生。在使用鼠標按動捲動列時,如果想使該捲動列獲得輸入焦點,那麼您必須將WS_TABSTOP標識符包含到CreateWindow呼叫的窗口類別參數中。當滾動條擁有輸入焦點時,在該滾動條的小方框上將顯示一個閃爍的灰色塊。
爲了給滾動條提供全面的鍵盤接口,還需要另外一些工作。首先,WndProc窗口消息處理程序必須使滾動條擁有輸入焦點,它是通過處理WM_SETFOCUS消息來完成這一點的,該WM_SETFOCUS消息是當滾動條獲得輸入焦點時其父窗口接收到的。WndProc給其中一個滾動條設定輸入焦點。
SetFocus (hwndScroll[idFocus]) ;
其中idFocus是一個整體變量。
但是,還需要一些藉助鍵盤尤其是Tab鍵,來從一個滾動條轉換到另一個滾動條的方法。這比較困難,因爲一旦某個滾動條擁有了輸入焦點,它就處理所有的鍵盤輸入,但滾動條只關心光標鍵,而忽略Tab鍵。解決這一兩難處境的方法是「窗口子類別化」。我們將用它來給COLORS1增加使用Tab鍵從一個滾動條跳到另一個滾動條的功能。
窗口子類別化(Window Subclassing)
滾動條控件的窗口消息處理程序是Windows內部的。但是,將GWL_WNDPROC標識符作爲參數來呼叫GetWindowLong,您就可以得到這個窗口消息處理程序的地址。另外,您可以呼叫SetWindowLong給該滾動條設定一個新的窗口消息處理程序,這個技術叫做「窗口子類別化」,非常有用。它能讓您給現存的窗口消息處理程序設定「掛勾」,以便在自己的程序中處理一些消息,同時將其它所有消息傳遞給舊的窗口消息處理程序。
在COLORS1中對捲動消息進行初步處理的窗口消息處理程序叫做ScrollProc,它在COLORS1.C文件的尾部。由於ScrollProc是COLORS1中的函數,而Windows將呼叫COLORS1,所以ScrollProc必須被定義爲callback函數。
對三個滾動條中的每一個,COLORS1使用SetWindowLong來設定新的滾動條窗口消息處理程序的地址,並取得現存滾動條窗口消息處理程序的地址:
OldScroll[i] = (WNDPROC) SetWindowLong (hwndScroll[i], GWL_WNDPROC,
(LONG) ScrollProc)) ;
現在,函數ScrollProc得到了Windows發送到COLORS1中三個滾動條(當然不是其它程序中的滾動條)的滾動條窗口消息處理程序的全部消息。ScrollProc窗口消息處理程序在接收到Tab或者Shift-Tab鍵時,就將輸入焦點改變到下一個(或者上一個)滾動條。它使用CallWindowProc呼叫舊的滾動條窗口消息處理程序。
給背景着色
當COLORS1定義它的窗口類別時,也爲其顯示區域背景定義了一個實心的黑色畫刷:
wndclass.hbrBackground = CreateSolidBrush (0) ;
當您改變COLORS1的滾動條設定時,程序必須建立一個新的畫刷,並將該新畫刷句柄放入窗口類別結構中。如同使用GetWindowLong和SetWindowLong能得到並設定滾動條窗口消息處理程序一樣,用GetClassWord和SetClassWord能得到這個畫刷的句柄。
您可以建立新的畫刷並將其句柄插入窗口類別結構中,然後刪除舊的畫刷:
DeleteObject ((HBRUSH)
SetClassLong (hwnd, GCL_HBRBACKGROUND, (LONG)
CreateSolidBrush (RGB (color[0], color[1], color[2])))) ;
Windows下一次重新爲窗口的背景着色時,將使用這個新畫刷。爲了強迫Windows抹掉背景,我們將使整個顯示區域無效:
InvalidateRect (hwnd, &rcColor, TRUE) ;
TRUE(非零)值作爲第三個參數,表示希望在重新着色之前刪去背景。
InvalidateRect使Windows在窗口消息處理程序的消息隊列中放進一個WM_PAINT消息。由於WM_PAINT消息的優先等級比較低,所以,如果您還在使用鼠標或者光標鍵移動滾動條的話,這個消息將不會立即被處理。如果您想在顏色改變之後使該窗口立即變成最新的(目前的),那麼您可以在InvalidateRect之後增加下面的敘述:
UpdateWindow (hwnd) ;
但這會使得鍵盤和鼠標處理變慢。
COLORS1中的WndProc函數不處理WM_PAINT消息,而是將其傳給DefWindowProc。Windows對WM_PAINT消息的內定處理只是呼叫BeginPaint和EndPaint使窗口生效。因爲在InvalidateRect呼叫中已經指定背景要被抹掉,所以BeginPaint呼叫使Windows發出一個WM_ERASEBKGND(刪除背景)消息,WndProc也將忽略這個消息。Windows用窗口類別中指定的畫刷將顯示區域的背景抹去,這樣就處理了這個消息。
在終止以前進行清除總是一個好主意,因此在處理WM_DESTROY消息處理期間,再一次呼叫DeleteObject:
DeleteObject ((HBRUSH)
SetClassLong (hwnd, GCL_HBRBACKGROUND,
(LONG) GetStockObject (WHITE_BRUSH))) ;
給滾動條和靜態文字着色
在COLORS1中,三個滾動條的內部和六個文字字段中的文字着色爲紅、綠和藍色。滾動條的着色是通過處理WM_CTLCOLORSCROLLBAR消息來完成的。
在WndProc中,我們爲畫刷定義了一個由三個句柄組成的靜態數組:
static HBRUSH hBrush [3] ;
在處理WM_CREATE期間,我們建立三個畫刷:
for (I = 0 ; I < 3 ; I++)
hBrush[0] = CreateSolidBrush (crPrim [I]) ;
其中crPrim數組中包含三種原色的RGB值。在WM_CTLCOLORSCROLLBAR處理期間窗口消息處理程序傳回這三畫刷中的一個:
case WM_CTLCOLORSCROLLBAR:
i = GetWindowLong ((HWND) lParam, GWL_ID) ;
return (LRESULT) hBrush [i] ;
在處理WM_DESTROY消息的過程中,這些畫刷必須被刪除:
for (i = 0 ; i < 3 ; i++)
DeleteObject (hBrush [i])) ;
同樣地,靜態文字字段中的文字是在處理WM_CTLCOLORSTATIC消息中呼叫SetTextColor來着色的。文字背景用SetBkColor函數設定爲系統顏色COLOR_BTNHIGHLIGHT,這導致文字背景顏色和滾動條與文字後面的靜態矩形控件的顏色一樣。對於靜態文字控件,這種文字背景顏色只用於字符串中每個字符後面的矩形,而不會用於整個控件窗口。爲了實作這一點,窗口消息處理程序還必須傳回COLOR_BTNHIGHLIGHT顏色畫刷的句柄。這個畫刷被稱爲hBrushStatic,它在WM_CREATE消息處理期間建立,在WM_DESTROY消息處理期間清除。
在WM_CREATE消息處理期間依據COLOR_BTNHIGHLIGHT顏色建立畫刷,並且在執行期間使用這一畫刷時,我們遇到了一個小問題。如果程序在執行期間改變了COLOR_BTNHIGHLIGHT顏色,那麼靜態矩形的顏色將發生變化,並且文字背景的顏色也會變化,但是文字窗口控件的整個背景將保持原有的COLOR_BTNHIGHLIGHT顏色。
爲了解決這個問題,COLORS1也簡單地通過使用新顏色重新建立hBrushStatic來處理WM_SYSCOLORCHANGE消息。