数学建模社区-数学中国

标题: C++ Builder 中的自画功能(上) [打印本页]

作者: 韩冰    时间: 2005-1-26 01:35
标题: C++ Builder 中的自画功能(上)
<>也许你都看见过一些列表框啦、下拉框啦什么的,他们的选项不是文字而是图形。比如像Word 97中设置字体的对话框里有一项“下划线”设置,其选项中就是各种线形,给人非常直观的感觉。这种下拉框在C++ Builder中能做出来吗?+ a0 x( @* `0 {# D5 n8 B
    Of course!我在自己的程序中,就成功地使用了这种列表框完成了线形的设置。OK,下面我们就来实际做一个程序。这个程序的作用是这样的:使用下拉列表框选择一种颜色,并把Label1的文字设置成该颜色。
  A  e; X% f/ I/ V' ~0 b3 Q9 a    先在Form上放一个ComboBox,取名为ComboBox1。把(注意是关键哦!)Style改为csOwnerDrawFixed,注意这一步很重要,因为如果不改成OwerDraw形式的话,就不会触发后面要用到的OnDrawItem事件。另外,为了编程方便,我们定义一个颜色数组(简单起见,就用五种颜色):
% |$ l& B" S( _0 k, ]  const TColor colors[5] = {clBlack, clWhite, clRed, clBlue, clGreen};
: k8 f; I; M3 c: J6 |, D再在ComboBox的Items里写上5行:黑色、白色、红色、兰色、绿色。
1 Z. v, h- D8 M. N8 l# ^最后在ComboBox的OnDrawItem里写上如下代码:</P>
$ G% e9 k+ s2 p- H% E<>void __fastcall TForm1::ComboBox1DrawItem(TWinControl *Control, int Index,# B3 `; ?( P% T2 Z% n( |
      TRect &amp;Rect, TOwnerDrawState State)" L: |0 s# F* A1 [' t# b6 \
//Control是需要自画的控件,Index是项数索引(从0开始),Rect是该项的绘画范围,State表示状态8 p+ _" G0 f( g8 Z: ~
//(选中、聚焦、无效等)
7 @! d' j1 Z& S) K4 K{
, k* H" W0 u0 m  b7 P5 I7 f6 p# m9 R  //画颜色索引的小方块: b% |4 Z" [0 z+ k# l, A; ^$ E5 y) V0 N
  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Color = colors[Index];9 P2 J# g4 H' P1 n  I) o
  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Style = bsSolid;
' f$ O- ~/ U! k7 l$ z/ Z. U1 b  ((TComboBox *)Control)-&gt;Canvas-&gt;FillRect(Classes::Rect(Rect.Left+2, Rect.Top+2, Rect.Left+14, Rect.Bottom-2));  //之所以用Classes::Rect是因为与参数重名
# X+ b  h# e" K* [  ((TComboBox *)Control)-&gt;Canvas-&gten-&gt;Color = clBlack;
6 M0 W( g4 T6 Z$ g2 t4 [  ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left+1, Rect.Top+1, Rect.Left+15, Rect.Bottom-1);</P>
' s& m! z) s5 k& z6 M! d) P8 V9 h<>  //写文字2 r: D9 U) q2 Z& z% Q9 i: Z
  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Color = clBlack;
  v8 \  ], A5 v6 U+ M: L9 @  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Name  = "宋体";# }. K: `! w" X
  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Size  = 9;</P>
+ p: [' t+ ^- n. b0 T5 x5 e<>  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Style = bsClear;
; h" N5 p  R; |  ((TComboBox *)Control)-&gt;Canvas-&gt;TextOut(Rect.Left+20, Rect.Top+2, ((TComboBox *)Control)-&gt;Items-&gt;Strings[Index]);</P>
0 ^7 ^9 V5 d+ A' _7 x) v<>  //标记当前鼠标所在项(焦点项)[本例中在外边画个蓝方框表示]1 A' i5 A8 q8 D" f+ Z
  if(State.Contains(odFocused))
/ S" Z- c+ q0 v6 J1 p8 C  {) v6 P9 P: P; O3 ~
    ((TComboBox *)Control)-&gt;Canvas-&gten-&gt;Color = clBlue;
* s# C6 a3 P6 a2 x& U! L    ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left, Rect.Top, Rect.Right, Rect.Bottom);4 A9 N# ]; }1 t1 l/ y) `- U* G
  }
' O$ P" T6 G- O  else7 K! Y  o/ n1 J3 P* b% g; [, z$ P
  {, d) b$ E+ e2 s8 L# {3 h
    ((TComboBox *)Control)-&gt;Canvas-&gten-&gt;Color = clWhite;
2 i& l2 N3 _' M1 R9 p    ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left, Rect.Top, Rect.Right, Rect.Bottom);; S' C0 {6 O6 {; [# e6 f, l
  }4 O  e# n  V! ?% n' R6 S1 b
}</P>
  L" P/ F9 ?& Z& d5 k<>好了,至此,一个自画的列表框就完成了。接下来的事情就非常简单了:
8 I$ x) Z8 P/ K8 m" A# c$ f在ComboBox的OnChange事件中写以下代码:</P>
9 Y8 R1 J" `$ {<>void __fastcall TForm1::ComboBox1Change(TObject *Sender)
( n' B4 h2 o4 F; P$ v{" k* w2 a* F2 I* y  L( u* n2 B2 ^
  Label1-&gt;Font-&gt;Color = colors[ComboBox1-&gt;ItemIndex];        $ K' ?, H; x8 E: n
}</P>
, z/ S9 O' m* \1 y<>就完成了Label字体变色的功能,我想不用解释了吧?' M* V8 D2 y/ e$ Y
运行看看效果吧!</P>9 F  Z' ?% A$ F$ D1 h
<>不过,现在在选择之前列表框中并没有东西,有时可能希望开始的时候列表框中有一个缺省选项,这时,只要在初始化时(比如FormCreate事件中),写上
: T- j4 \( Z" c, l5 [  ComboBox1-&gt;ItemIndex = 0;  //0 表示第一项
% v6 E" f/ t- @/ M+ ?- |0 c就行了。</P>
: X# g9 v( Z1 h+ \. u8 S6 c) x<>另外,你可能会注意到ComboBox的Style属性中有两个选项跟OwnerDraw有关,除了我们演示的这个OwnerDrawFixed之外,还有一个OwnerDrawVariable。它们有什么区别吗?4 D* _3 K' @4 l: g" c5 H" ^! {
当然啦,Fix指的是“固定”,Variable指的是“可变”。所以,如果你要画的每一项都是同样大小(就像我们在上面演示的那种),用Fix;否则,用Variable。但是使用Variable时,你在画之前,还要通过OnMeasureItem事件确定每一项的高度值。由于设置很简单,我就不演示了。</P>6 R1 ^- X/ k) Z: P4 R5 {
<>除了ComboBox和ListBox之外,PageControl(多页控件)的Tab也是可以自己画的。方法嘛差不多。我想就不罗嗦了吧。但是同样要注意要把OwnerDraw设成true。</P>




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5