GDI+では、パスに文字列のパスを追加する事ができます。このパスは 文字列を構成する頂点の情報となるので、追加したパスを描画する事によ って縁を描画する事ができます。
#include <stdio.h> #include <tchar.h> #include <locale.h> #include <string> #include <windows.h> #include <wingdi.h> #include <gdiplus.h> // ライブラリ #pragma comment( lib, "gdiplus.lib" ) /* GDI+による縁取り文字列の描画 */ void DrawBorderingString ( Gdiplus::Graphics& oGraphics , Gdiplus::RectF* opRect , Gdiplus::Pen* opPen , std::wstring oFontFamilyName , std::wstring oString , Gdiplus::REAL oEmSize ) { // フォントファミリー Gdiplus::FontFamily oFontFamily( oFontFamilyName.c_str() ); // パス Gdiplus::GraphicsPath oGraphicsPath; // 文字列パスの追加 oGraphicsPath.AddString( oString.c_str() , -1 , &oFontFamily , Gdiplus::FontStyleBold , oEmSize , *opRect , Gdiplus::StringFormat::GenericDefault() ); // パスの描画 oGraphics.DrawPath( opPen, &oGraphicsPath ); } /* メインウインドウイベント処理 */ LRESULT CALLBACK eMainWindowProc ( HWND hWnd // handle to window , UINT uMsg // message identifier , WPARAM wParam // first message parameter , LPARAM lParam // second message parameter ) { switch( uMsg ) { case WM_CREATE: //-------------------------------------------- // WM_CREATE //-------------------------------------------- { CREATESTRUCT* tpCreateSt = (CREATESTRUCT*)lParam; // ウインドウを表示する ::ShowWindow( hWnd, SW_SHOW ); } break; case WM_DESTROY: //-------------------------------------------- // WM_DESTROY //-------------------------------------------- { // 終了する( 引数はそのまま終了コードとなります ) ::PostQuitMessage( 0 ); } break; case WM_PAINT: //-------------------------------------------- // WM_PAINT //-------------------------------------------- { PAINTSTRUCT tPaintStruct; // 描画開始 HDC hDC = ::BeginPaint( hWnd, &tPaintStruct ); { Gdiplus::Graphics oGraphics( hDC ); /* GDI+による縁取り文字列の描画 */ { Gdiplus::Pen oPen( Gdiplus::Color( 255, 0, 0 ), 3 ); Gdiplus::RectF oRectF( 10, 10, 600, 420 ); // GDI+による縁取り文字列の描画 DrawBorderingString( oGraphics , &oRectF , &oPen , L"MS 明朝" , L"あいうえおかきくけこさしすせそたちつてと" , 100 ); } } // 描画終了 ::EndPaint( hWnd, &tPaintStruct ); } return( FALSE ); } // デフォルト処理呼び出し return ::DefWindowProc( hWnd, uMsg, wParam, lParam ); } /* GDI+による縁取り文字列の描画 */ int _tmain ( int argc , _TCHAR* argv[] ) { // 標準出力にユニコードを表示できるようにする setlocale( LC_ALL, "Japanese" ); WNDCLASSEX tWndClass; HINSTANCE hInstance; TCHAR* cpClassName; TCHAR* cpWindowName; TCHAR* cpMenu; HWND hWnd; MSG tMsg; // GDI+の開始 ULONG_PTR gdiplusToken; Gdiplus::GdiplusStartupInput gdiplusStartupInput; Gdiplus::GdiplusStartup( &gdiplusToken, &gdiplusStartupInput, 0 ); // アプリケーションインスタンス hInstance = ::GetModuleHandle( NULL ); // クラス名称 cpClassName = _T("MainWindowClass"); // メニュー cpMenu = MAKEINTRESOURCE( NULL ); // ウインドウ名称 cpWindowName = _T("GDI+による縁取り文字列の描画"); // ウインドウクラスパラメータセット tWndClass.cbSize = sizeof( WNDCLASSEX ); tWndClass.style = CS_HREDRAW | CS_VREDRAW; tWndClass.lpfnWndProc = eMainWindowProc; tWndClass.cbClsExtra = 0; // ::GetClassLong で取得可能なメモリ tWndClass.cbWndExtra = 0; // ::GetWindowLong で取得可能なメモリ tWndClass.hInstance = hInstance; tWndClass.hIcon = ::LoadIcon( NULL, IDI_APPLICATION ); tWndClass.hCursor = ::LoadCursor( NULL, IDC_ARROW ); tWndClass.hbrBackground = (HBRUSH)( COLOR_WINDOW + 1 ); tWndClass.lpszMenuName = cpMenu; tWndClass.lpszClassName = cpClassName; tWndClass.hIconSm = NULL; // ウインドウクラス生成 if ( 0 == ::RegisterClassEx( &tWndClass ) ) { /* 失敗 */ return( -1 ); } // ウインドウを生成する hWnd = ::CreateWindowEx ( 0 // extended window style , tWndClass.lpszClassName // pointer to registered class name , cpWindowName // pointer to window name , WS_OVERLAPPEDWINDOW // window style , CW_USEDEFAULT // horizontal position of window , CW_USEDEFAULT // vertical position of window , 640 // window width , 480 // window height , NULL // handle to parent or owner window , NULL // handle to menu, or child-window identifier , hInstance // handle to application instance , (VOID*)0x12345678 // pointer to window-creation data ); /* メッセージループ */ while( 0 != ::GetMessage( &tMsg, NULL, 0, 0 ) ) { ::TranslateMessage ( &tMsg ); ::DispatchMessage ( &tMsg ); } // GDI+の終了 Gdiplus::GdiplusShutdown( gdiplusToken ); // WM_QUITの終了コードを返却する return( tMsg.wParam ); }