韩冰 发表于 2005-1-26 01:35

C++ Builder 中的自画功能(上)

<P>也许你都看见过一些列表框啦、下拉框啦什么的,他们的选项不是文字而是图形。比如像Word 97中设置字体的对话框里有一项“下划线”设置,其选项中就是各种线形,给人非常直观的感觉。这种下拉框在C++ Builder中能做出来吗?
    Of course!我在自己的程序中,就成功地使用了这种列表框完成了线形的设置。OK,下面我们就来实际做一个程序。这个程序的作用是这样的:使用下拉列表框选择一种颜色,并把Label1的文字设置成该颜色。
    先在Form上放一个ComboBox,取名为ComboBox1。把(注意是关键哦!)Style改为csOwnerDrawFixed,注意这一步很重要,因为如果不改成OwerDraw形式的话,就不会触发后面要用到的OnDrawItem事件。另外,为了编程方便,我们定义一个颜色数组(简单起见,就用五种颜色):
  const TColor colors = {clBlack, clWhite, clRed, clBlue, clGreen};
再在ComboBox的Items里写上5行:黑色、白色、红色、兰色、绿色。
最后在ComboBox的OnDrawItem里写上如下代码:</P>
<P>void __fastcall TForm1::ComboBox1DrawItem(TWinControl *Control, int Index,
      TRect &amp;Rect, TOwnerDrawState State)
//Control是需要自画的控件,Index是项数索引(从0开始),Rect是该项的绘画范围,State表示状态
//(选中、聚焦、无效等)
{
  //画颜色索引的小方块
  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Color = colors;
  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Style = bsSolid;
  ((TComboBox *)Control)-&gt;Canvas-&gt;FillRect(Classes::Rect(Rect.Left+2, Rect.Top+2, Rect.Left+14, Rect.Bottom-2));  //之所以用Classes::Rect是因为与参数重名
  ((TComboBox *)Control)-&gt;Canvas-&gt;Pen-&gt;Color = clBlack;
  ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left+1, Rect.Top+1, Rect.Left+15, Rect.Bottom-1);</P>
<P>  //写文字
  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Color = clBlack;
  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Name  = "宋体";
  ((TComboBox *)Control)-&gt;Canvas-&gt;Font-&gt;Size  = 9;</P>
<P>  ((TComboBox *)Control)-&gt;Canvas-&gt;Brush-&gt;Style = bsClear;
  ((TComboBox *)Control)-&gt;Canvas-&gt;TextOut(Rect.Left+20, Rect.Top+2, ((TComboBox *)Control)-&gt;Items-&gt;Strings);</P>
<P>  //标记当前鼠标所在项(焦点项)[本例中在外边画个蓝方框表示]
  if(State.Contains(odFocused))
  {
    ((TComboBox *)Control)-&gt;Canvas-&gt;Pen-&gt;Color = clBlue;
    ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left, Rect.Top, Rect.Right, Rect.Bottom);
  }
  else
  {
    ((TComboBox *)Control)-&gt;Canvas-&gt;Pen-&gt;Color = clWhite;
    ((TComboBox *)Control)-&gt;Canvas-&gt;Rectangle(Rect.Left, Rect.Top, Rect.Right, Rect.Bottom);
  }
}</P>
<P>好了,至此,一个自画的列表框就完成了。接下来的事情就非常简单了:
在ComboBox的OnChange事件中写以下代码:</P>
<P>void __fastcall TForm1::ComboBox1Change(TObject *Sender)
{
  Label1-&gt;Font-&gt;Color = colors;        
}</P>
<P>就完成了Label字体变色的功能,我想不用解释了吧?
运行看看效果吧!</P>
<P>不过,现在在选择之前列表框中并没有东西,有时可能希望开始的时候列表框中有一个缺省选项,这时,只要在初始化时(比如FormCreate事件中),写上
  ComboBox1-&gt;ItemIndex = 0;  //0 表示第一项
就行了。</P>
<P>另外,你可能会注意到ComboBox的Style属性中有两个选项跟OwnerDraw有关,除了我们演示的这个OwnerDrawFixed之外,还有一个OwnerDrawVariable。它们有什么区别吗?
当然啦,Fix指的是“固定”,Variable指的是“可变”。所以,如果你要画的每一项都是同样大小(就像我们在上面演示的那种),用Fix;否则,用Variable。但是使用Variable时,你在画之前,还要通过OnMeasureItem事件确定每一项的高度值。由于设置很简单,我就不演示了。</P>
<P>除了ComboBox和ListBox之外,PageControl(多页控件)的Tab也是可以自己画的。方法嘛差不多。我想就不罗嗦了吧。但是同样要注意要把OwnerDraw设成true。</P>
页: [1]
查看完整版本: C++ Builder 中的自画功能(上)