在一個點對點文件傳輸的項目中,我需要顯示文件傳輸的實時信息:傳輸的文件列表和當前傳輸的文件,當時我想到了用ListBox,但是但我用了ListBox後,我發現它不能改變控件中文本想的顏色,於是我就想擴展一下ListBox控件------ListBoxEx。
我的目標是給空間加上圖標,還要能時時改變控件文本顏色。於是從ListBox派生類
public class ListBoxEx : ListBox {…}
為了操作方便我為ListBoxEx的每一項設計專門的類ListBoxExItem
public class ListBoxExItem {…}
為了保持我這個控件與WinForm的標准控件的操作借口一致,我又重新設計了兩個集合類:
public class ListBoxExItemCollection : IList, ICollection, IEnumerator {} //這個類相對於標准ListBox中的ObjectCollection,這個類作為ListBoxEx中的Items屬性的類型 public class SelectedListBoxExItemCollection : : IList, ICollection, IEnumerator{} //這個類相對於標准ListBox中的SelectedObjectCollection,這個類作為ListBoxEx中的SelectedItems屬性的類型
下面看兩個集合類的實現:
ListBoxExItemCollection的實現:為了做到對集合(Items)的操作能夠及時反映到ListBoxEx的控件中所以,此類只是對ListBox中Items(ObjectCollection類型)作了一層包裝,就是把ListBox中Items屬性的所有方法的只要是object類型的參數都轉換成ListBoxExItem,比如:
public void Remove(ListBoxExItem item) { this._Items.Remove(item); //_Items為ObjectCollection類型 } public void Insert(int index, ListBoxExItem item) { this._Items.Insert(index, item); } public int Add(ListBoxExItem item) { return this._Items.Add(item); }
由上可知,ListBoxExItemCollection中有一個構造函數來傳遞ListBox中的Items對象
private ObjectCollection _Items; public ListBoxExItemCollection(ObjectCollection baseItems) { this._Items = baseItems; }
而SelectedListBoxExItemCollection類的實現也用同樣的方法,只不過是對SelectedObjectCollection包裝罷了。
集合實現後,再來看ListBoxExItem的實現:
為了使它支持圖標和多種顏色添加如下成員
private int _ImageIndex; public int ImageIndex { get { return this._ImageIndex; } set { this._ImageIndex = value;} } private Color _ForeColor; public Color ForeColor { get{ return this._ForeColor;} set { this._ForeColor = value; this.Parent.Invalidate(); } }
當然還有:
private string _Text; public string Text { get { return this._Text; } set { this._Text = value; } }
為了控件能正確顯示此項的文本,還必須重寫ToString()方法
public override string ToString() { return this._Text; }
再看ListBoxEx的實現:
為了使控件能夠自我繪制,所以:DrawMode = DrawMode.OwnerDrawFixed;
為了覆蓋基類的Items等相關屬性添加
private ListBoxExItemCollection _Items; //在構造函數中創建
同時還需要重寫屬性Items:
new public ListBoxExItemCollection Items { get { return this._Items; } } new public ListBoxExItem SelectedItem //強制轉換為ListBoxExItem { get{ return base.SelectedItem as ListBoxExItem;} set{ base.SelectedItem = value;} } new public SelectedListBoxExItemCollection SelectedItems //重新包裝SelectedItems { get { return new SelectedListBoxExItemCollection(base.SelectedItems); } }
為了支持圖標,添加一個圖像列表imagelist
private ImageList imageList; public ImageList ImageList { get { return this.imageList; } set { this.imageList = value; this.Invalidate();//圖像列表改變後馬上更新控件 } }
而此控件的核心卻在一個方法OnDrawItem,這個方法每當控件的項需要重繪時就被調用
protected override void OnDrawItem(System.Windows.Forms.DrawItemEventArgs pe) { pe.DrawBackground(); //畫背景 pe.DrawFocusRectangle(); //畫邊框 Rectangle bounds = pe.Bounds; // Check whether the index is valid if(pe.Index >= 0 && pe.Index < base.Items.Count) { ListBoxExItem item = this.Items[pe.Index]; //取得需要繪制項的引用 int iOffset = 0; // If the image list is present and the image index is set, draw the image if(this.imageList != null) { if (item.ImageIndex > -1 && item.ImageIndex < this.imageList.Images.Count) { this.imageList.Draw(pe.Graphics, bounds.Left, bounds.Top, bounds.Height, bounds.Height, item.ImageIndex); //繪制圖標 } iOffset += bounds.Height;//this.imageList.ImageSize.Width; } // Draw item text pe.Graphics.DrawString(item.Text, pe.Font, new SolidBrush(item.ForeColor),bounds.Left + iOffset, bounds.Top); //根據項的顏色繪制文本 } base.OnDrawItem(pe); } }
到此為止,ListBoxEx以完整的實現,並且支持可視化設計。