南希系列中就用到了GDI+编程 5 P3 p2 }$ B4 v+ y! T! ~
5 d5 {0 X2 E; c! w- \+ D: V字体是文字显示和打印的外观形式,它包括了文字的字样、风格和尺寸等多方面的属性。适当地选用不同的字体,可以大大地丰富文字的外在表现力。例如,把文字中某些重要的字句用较粗的字体显示,能够体现出突出、强调的意图。当然,文本输出时还可使用其格式化属性和显示质量来优化文本显示的效果。 / P$ Q; C8 b# O- i; M7 b
4 i& w, x" y. D1 f, I
字体属性和字体创建 + i/ b1 `% y. w! R8 o
+ T7 l/ p& M. f7 W, c 字体的属性有很多,这里主要介绍字样、风格和尺寸三个主要属性。 $ J5 t! s" g9 o+ m2 F/ T
1 ^) @0 s7 `& _ 字样是字符书写和显示时表现出的特定模式,例如,对于汉字,通常有宋体、楷体、仿宋、黑体、隶书以及幼圆等多种字样。GDI+是通过FontFamily类来定义字样的,例如下面的代码: 4 G/ D( ?3 E# m0 ~& y2 H! O
0 Z/ n7 ~: R& i" z
FontFamily fontFamily(L"幼圆"); // 定义"幼圆"字样 . H/ x: P' h1 ^, F
0 |' u9 t7 J3 ^0 d& @' B* G. j 字体风格主要表现为字体的粗细和是否倾斜等特点。GDI+为用户提供了一些预定义的字体风格:FontStyleRegular(正常)、FontStyleBold(加粗)、FontStyleItalic(斜体)、FontStyleBoldItalic (粗斜体)、FontStyleUnderline(下划线)和FontStyleStrikeout(删除线)。 % V' ~3 U. v1 o5 ~( r
6 N( n9 F: H1 B+ V) l, C) i 字体尺寸是用来指定字符所占区域的大小,通常用字符高度来描述。字体尺寸可以取毫米或英寸作为单位,但为了直观起见,也常常采用一种称为点的单位,一点约折合为1/72英寸。对于汉字,还常用号数来表示字体尺寸,初号字最大,以下依次为小初、一号、小一、二号、小二??,如此类推,字体尺寸起来越小。GDI+为用户提供了UnitDisplay(1/75英寸)、UnitPixel(像素)、UnitPoint(点)、UnitInch(英寸)、UnitDocument(1/300英寸)、UnitMillimeter(毫米)等字体尺寸单位。
) y& N2 ?8 K4 u/ f; Z" y: H1 u, X/ L
使用GDI+中的Font类,可以直接通过构造函数创建一个字体对象,例如下列代码: 9 x! j) K) |& l
* v6 c& R5 T; q+ W+ ^& X2 hFont font(&fontFamily, 12, FontStyleRegular, UnitPoint);
1 J; s& f$ F& b! K! j5 \# N9 F3 `, x/ o( m, u
构造函数的第一个参数是用来指定FontFamily类对象指针,第二参数是用来指定字体的尺寸,它的实际大小取决于第四个参数所指定的尺寸单位。第三个参数用来指定字体风格。 ( c! ]. A4 [2 T2 o. u
* s) F, q) K# Z( v" t 为了与原来的GDI字体相兼容,Font的构造函数还有另外一种型式:
3 v4 }: F( _& r; i* H) ^1 g1 v0 S) Q# c4 U! q2 H
Font( HDC hdc, const LOGFONTW* logfont)
D" t& k6 n4 {4 v& T1 k/ @5 S( R* {
& [0 U& D( U2 b 其中,hdc是用来指定一个窗口的设备环境句柄,logfon是指向LOGFONT(逻辑字体)数据结构的指针。
9 P, \0 W: S9 D w$ ^- d
6 B6 j# w' p M7 P, d6 }文本输出
2 V/ {' P6 g- C, t4 C( F) ~& m6 p k3 }6 i5 _/ ?
文本的最终输出不仅依赖于文本的字体,而且还跟文本的颜色、对齐方式、字符间隔等有很大关系。GDI+只有一个输出文本的函数DrawString,它的原型如下: 5 \8 o: q" T8 W7 ~+ O, s% E
- z5 G* c0 _- J: e: w% i7 h" h
2 N* F; U/ i0 [ x+ B8 {+ ]* n% k* q
) b6 p0 Q$ I# P) C$ f# c. bDrawString( const WCHAR* string, INT length, const Font* font,
d: U# [" `& qconst RectF& layoutRect, const StringFormat* stringFormat, a$ H4 S) T3 }5 B
const Brush* brush ); ; g) N& I, v. N1 q) }) h$ X S+ `
2 \# r! m6 |: o8 X/ k2 ]% A
DrawString( const WCHAR* string, INT length, const Font* font, : s- s s* |( Y: ~" O
const PointF& origin, const Brush* brush );
0 b& ~5 G( S f
& \: ^; Y; `" k Z! WDrawString( const WCHAR* string, INT length, const Font* font, . O3 A1 d: Z- p" o
const PointF& origin, const StringFormat* stringFormat, & v: \8 m) W# {7 J' z
const Brush* brush);
Q% o* F1 h4 ]/ i$ \+ | ^% W
' L4 Z. e- k* P
0 \: F0 m* g" o$ U& k# |7 ]3 _# i' W
其中,string用来指定要输出的字符串,length表示该字符串的长度,font用来指定字体,layoutRect用来指定一个字符串所输出的矩形区域,stringFormat用来指定文本输出格式化属性,origin用来指定字符串输出的起点。需要注意的是,PointF和RectF类与Point和Rect类基本相同,所不同的是数据类型是浮点而后者是INT型。brush用来指定一个画刷,这个画刷既可以是SolidBrush和HatchBrush,也可以是TextureBrush(纹理画刷),甚至是渐变画刷。例如下面的代码,结果如图7.14所示。
& G4 X$ |! G. d2 l" r, R
' d8 M2 t( H1 U# v5 v4 d" z- vGraphics graphics( pDC->m_hDC );
+ q0 Y/ W( Y. h* ^& M" X4 _
/ M4 o+ Q$ P& Z a5 Q* w2 m0 OFontFamily fontFamily(L"幼圆");
% m% ` Y0 c# Q4 W! N ~Font font(&fontFamily, 20, FontStyleRegular, UnitPoint);
3 |0 A" _$ k3 Y# M, I. p/ bPointF pointF(30, 10);
5 J/ J, d6 S1 m3 x# \! rImage image(L"image.jpg");
& d- h2 r. H( f5 A/ y6 gTextureBrush tBrush(&image); ; w$ C3 X% {" `+ Y4 {
* A n4 A& ^, v$ ^8 ?: m. yLinearGradientBrush linGrBrush(
) D9 b% [6 H& B5 C4 J2 O! APoint(30, 50), 6 D" r: B$ ?: [7 [& b, x* r
Point(100, 50), . e. ~1 X$ a& ^1 L
Color(255, 255, 0, 0), 1 b6 X! q" W" K5 l7 I. t
Color(255, 0, 0, 255));
7 l! C) O3 O: F
- i( R! { }" bWCHAR string[256];
) Z$ a1 r) w1 m% g1 n, Iwcscpy(string, L"欢迎使用GDI+!");
$ Y" `& U. h, U5 m" F5 t& J' }/ e5 r0 C/ Q* ^
graphics.DrawString(string, (INT)wcslen(string), &font, pointF, &tBrush); 7 U: c* S' e& e0 ]1 ?
pointF.Y += 50; # y0 [8 P C8 C4 e, g
graphics.DrawString(string, (INT)wcslen(string), &font, pointF, 0 @% H) ^ Q6 O* E+ y. t0 G/ z& X
&linGrBrush); : K/ M$ c* Y, `
6 z/ j) Z2 \3 I, b8 m. W
需要说明的是,在GDI+中,我们可以通过SetTextRenderingHint来控制文本输出的质量。例如下面的代码,其结果如图7.15所示。
6 J0 J7 U9 V' r6 Y# n n/ X+ t" [$ b, V
Graphics graphics( pDC->m_hDC );
9 k1 y! n" z" z* G! O; i0 E
; a; Z1 V4 H w5 O" xFontFamily fontFamily(L"楷体_GB2312");
# |8 M. E* u5 uFont font(&fontFamily, 30, FontStyleRegular, UnitPixel); ' ^+ {* B, O. T* c
SolidBrush solidBrush(Color(255, 0, 0, 255)); , X5 J* ^4 J5 c8 ~- O# S, K
WCHAR string1[] = L"没有任何优化处理"; 8 `9 c2 {& R( v! B+ A
WCHAR string2[] = L"字体优化,但边不作平滑处理"; 6 `# A7 C: D: @1 z
WCHAR string3[] = L"消除走样,且边作平滑处理";
' J; j0 M. E0 M
: Q B6 T1 d# t) Y7 C/ \! r( vgraphics.SetTextRenderingHint(TextRenderingHintSingleBitPerPixel);
- O( E3 P% ^7 C: _! O6 ]) Ygraphics.DrawString( . s3 J A: s( ^( o+ k
string1, (INT)wcslen(string1), &font, PointF(10, 10), &solidBrush);
* ~7 D6 o# h$ W9 w X% H$ r: \
5 A/ L$ T8 m+ j$ egraphics.SetTextRenderingHint(TextRenderingHintSingleBitPerPixelGridFit);
. `4 u6 X; ]% ~! |2 N) ~% P. ngraphics.DrawString( , X; d" W2 r4 M
string2, (INT)wcslen(string2), &font, PointF(10, 50), &solidBrush); ( ?; |- i0 I) q5 i0 o) Q6 u
4 P; J d! y* m: `, g( W! J
graphics.SetTextRenderingHint(TextRenderingHintAntiAliasGridFit);
0 X+ c& h8 e) |( W% |6 e2 R. a+ agraphics.DrawString(
/ n3 [. [5 v* a& l5 t% estring3, (INT)wcslen(string3), &font, PointF(10, 90), &solidBrush); 4 Y' i; ^4 s6 H+ @
0 D" z0 g) s- ]) w7 i6 [ 3 l0 i. ~/ l8 ^8 ?: Z
4 `9 c" _$ y8 G2 B5 i& S ^
文本格式化属性
; m! z% S7 H' X `2 |4 U; G J4 M
: i+ T. p0 w; W2 ` _& c 文本的格式属性通常包括对齐方式、字符间隔以及文本调整等。GDI+提供StringFormat类来控制这些格式属性,通常我们调用以下几个函数来进行相关属性设置。 . B9 l! `- O1 j- }% y* v
7 w2 g4 w* V+ X- `Status SetAlignment( StringAlignment align); 0 N- m4 O- `! a! P. T
Status SetLineAlignment( StringAlignment align);
" p8 t" C+ J8 r, k1 g* A9 g% P7 s/ }# v r# M T' ~. c/ D
该函数用来设置文本对齐方式,align可以是StringAlignmentNear(左对齐或右对齐,取决于书写方向是从左到右还是从右到左)、StringAlignmentCenter(居中)或StringAlignmentFar(两端对齐)。 # Y+ L+ a! b0 \
5 u" t; r0 E0 ]/ C8 }' z% g$ o
Status SetFormatFlags( INT flags );
& Z1 b- q9 \9 e Q) ^4 O+ T" B3 ~
. [. _6 n4 }8 V 该函数用来设置文本格式化标志,flags可以是StringFormatFlagsDirectionRightToLeft(水平阅读方向是从右向左)和StringFormatFlagsDirectionVertical(垂直文本)等值。例如下面的代码,其结果如图7.16所示。 ) S0 Y$ c- ~# i! I2 d
) Q2 v' q9 m2 s' rGraphics graphics( pDC->m_hDC ); + l. V& y- o* h+ V
SolidBrush solidBrush(Color::Blue); $ R- d) e) ?" R6 t" o
FontFamily fontFamily(L"楷体_GB2312"); * m$ K% O1 C7 X$ }
Font font(&fontFamily, 16, FontStyleRegular, UnitPoint);
6 k q! {* t# F. X' g7 _
; X+ G1 T2 I: `) U' N+ X8 L* G: o* pStringFormat stringFormat;
; i, y% G4 h* u3 G% GstringFormat.SetFormatFlags( StringFormatFlagsDirectionRightToLeft |
9 X, Y/ f/ q% Y+ ?6 I, S3 C; E QStringFormatFlagsDirectionVertical | $ Y% V3 c! } ]2 J0 \
StringFormatFlagsNoFitBlackBox); # o8 f0 b0 V* f- B' E, S- e
stringFormat.SetAlignment(StringAlignmentCenter); , H) M9 G2 g; F
WCHAR string[] = L"这些文字是垂直居中且是从右到左的阅读次序, 它们是通过 \
2 \( c5 w$ u$ H" ]. XSetFormatFlags和SetAlignment来设置的!"; 9 {* q' K, U9 D% X
# k1 ^) O- l; {, R7 W2 A& M
graphics.DrawString( string, (INT)wcslen(string), &font, 9 y+ E9 z7 y; f! A' k& ]
RectF(30, 30, 150, 200), &stringFormat, &solidBrush); |