WTLではGDIオブジェクトのブラシをCBrushTというクラステンプレートでカプセル化しています。
テンプレート引数はbool値で、trueの場合はデストラクタでDeleteObject()を呼び出し、
falseの場合は呼び出しません。atlgdi.hヘッダではtypedefによって次のように宣言されています。
// atlgdi.h
typedef CBrushT<false> CBrushHandle;
typedef CBrushT<true> CBrush;
|
WTLのCBrushクラスは、MFCの同名のクラスとほぼ同じメンバ関数を用意しています。
以下に示すのは、CBrushクラスを使用してクライアント矩形に接する青い楕円を描画する例です。
// stdafx.h
#pragma once
#define WINVER _WIN32_WCE
#define _SECURE_ATL 1
#include <atlbase.h>
#include <atlapp.h>
extern CAppModule _Module;
#include <atlwin.h>
#include <aygshell.h>
#pragma comment(lib, "aygshell.lib")
#include <atlcrack.h>
#include <atlmisc.h> // CRectを使用するため
|
// MainWindow.h
#pragma once
class CMainWindow : public CWindowImpl<CMainWindow>,
public CMessageFilter, public CIdleHandler
{
public:
DECLARE_WND_CLASS(_T("SampleProject"));
virtual BOOL PreTranslateMessage(MSG* pMsg){
return FALSE;
}
virtual BOOL OnIdle(){
return FALSE;
}
BEGIN_MSG_MAP(CMainWindow)
MSG_WM_PAINT(OnPaint)
MSG_WM_ACTIVATE(OnActivate)
MSG_WM_SETTINGCHANGE(OnSettingChange)
MSG_WM_CREATE(OnCreate)
MSG_WM_DESTROY(OnDestroy)
END_MSG_MAP()
void OnPaint(CDCHandle /*dc*/){
CPaintDC dc(m_hWnd);
// 青いブラシを作成し、選択
CBrush brush;
brush.CreateSolidBrush(RGB(0, 0, 255));
HBRUSH hOldBrush = dc.SelectBrush(brush);
// クライアント矩形を取得し、円を描画
CRect rect;
GetClientRect(rect);
dc.Ellipse(rect);
// 元のブラシを選択
dc.SelectBrush(hOldBrush);
}
void OnActivate(UINT nState, BOOL bMinimized, CWindow wndOther){
SHFullScreen(m_hWnd, SHFS_HIDESIPBUTTON);
}
void OnSettingChange(UINT uFlags, LPCTSTR lpszSection){
SHFullScreen(m_hWnd, SHFS_HIDESIPBUTTON);
}
int OnCreate(LPCREATESTRUCT lpCreateStruct){
// メッセージループにメッセージフィルタとアイドルハンドラを追加
CMessageLoop* pLoop = _Module.GetMessageLoop();
pLoop->AddMessageFilter(this);
pLoop->AddIdleHandler(this);
return 0;
}
void OnDestroy(){
// メッセージループからメッセージフィルタとアイドルハンドラを削除
CMessageLoop* pLoop = _Module.GetMessageLoop();
pLoop->RemoveMessageFilter(this);
pLoop->RemoveIdleHandler(this);
PostQuitMessage(0);
}
};
|
// SampleProject.cpp
#include "stdafx.h"
#include "MainWindow.h"
CAppModule _Module;
int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
TCHAR szAppName[] = _T("SampleProject");
HWND hWnd = FindWindow(szAppName, szAppName);
if(hWnd){
SetForegroundWindow((HWND)((ULONG) hWnd | 0x00000001));
return 0;
}
_Module.Init(NULL, hInstance);
CMessageLoop theLoop;
_Module.AddMessageLoop(&theLoop);
CMainWindow wnd;
wnd.Create(NULL, CWindow::rcDefault, szAppName, WS_OVERLAPPEDWINDOW | WS_VISIBLE);
int nRet = theLoop.Run();
_Module.RemoveMessageLoop();
_Module.Term();
return nRet;
}
|
WM_PAINTメッセージハンドラで、まずブラシオブジェクトを作成し、
それをSelectBrush()で現在のブラシとして選択します。
SelectBrush()は引数にHBRUSH型の変数を受け取りますが、
CBrushTクラステンプレートはHBRUSH変換演算子を持っているのでそのまま渡すことができます。
SelectBrush()は以前のブラシハンドルを返すので、
これをhOldBrushという変数に保存しておきます。
次に、円を描画します。青い円が描画されます。
最後に、先ほど保存しておいた元のブラシハンドルに戻すため、
再度SelectBrush()を呼び出して元のブラシハンドルを選択します。
ところで、WTLはストックブラシもサポートしています。
以下に示すのは、WM_PAINTメッセージハンドラでストックブラシを使用して黒く塗りつぶされた円を描画する例です。
// WM_PAINTメッセージハンドラ
void OnPaint(CDCHandle /*dc*/){
CPaintDC dc(m_hWnd);
// ストックブラシを選択
HBRUSH hOldBrush = dc.SelectStockBrush(BLACK_BRUSH);
// クライアント矩形を取得し、円を描画
CRect rect;
GetClientRect(rect);
dc.Ellipse(rect);
// 元のブラシを選択
dc.SelectBrush(hOldBrush);
}
|
また、atlmisc.hヘッダに定義されているAtlGetStockBrush()というグローバル関数を使えば、
ストックブラシのハンドルを取得することができます。
// ストックブラシのハンドルを取得
HBRUSH hStockBrush = AtlGetStockBrush(BLACK_BRUSH);
|
|