.NET WinForm程序中给DataGridView表头添加下拉列表实现数据过滤
轉(zhuǎn):http://www.cnblogs.com/jaxu/archive/2011/08/04/2127365.html
我們見過Excel中的數(shù)據(jù)過濾功能,可以通過點(diǎn)擊表頭上的下拉列表來實(shí)現(xiàn)數(shù)據(jù)的過濾,這個(gè)功能很實(shí)用,省去了我們需要在程序中單獨(dú)設(shè)計(jì)數(shù)據(jù)的查詢過濾模塊,功能直接依賴于數(shù)據(jù)綁定控件DataGridView。先來看看Excel中的數(shù)據(jù)過濾功能。
要想在DataGridView中實(shí)現(xiàn)類似于Excel的這種功能其實(shí)也并非難事。來看看msdn上的一篇文章,上面有詳細(xì)的介紹,不過目前只有全英文的版本。http://msdn.microsoft.com/en-us/library/aa480727.aspx。里面提供的下載示例我這里也可以提供一份:DataGridViewAutoFilter.zip
文章講了很多有關(guān)如何實(shí)現(xiàn)數(shù)據(jù)過濾的知識(shí),如果你有耐心可以通讀一遍,應(yīng)該能有不小的收獲。其實(shí)這里面的原理就是我們需要自定義一種DataGridViewColumn,它能支持用戶通過點(diǎn)擊表頭上的下拉列表來實(shí)現(xiàn)DataGridView的數(shù)據(jù)過濾。自定義的DataGridViewColumn可以繼承自現(xiàn)有的DataGridViewTextBoxColumn類型,另外還需要自定義一個(gè)繼承自DataGridViewColumnHeaderCell的類型,它負(fù)責(zé)在DataGridView表頭上呈現(xiàn)一個(gè)下拉列表,并完成數(shù)據(jù)過濾的選擇功能。下載上面的DataGridViewAutoFilter.zip壓縮包,將里面對應(yīng)編程語言中的DataGridViewAutoFilterColumnHeaderCell.cs和DataGridAutoFilterTextBoxColumn.cs兩個(gè)文件加入到你的工程中。然后需要重新定義DataGridView中的列,如果你是手動(dòng)指定DataGridView的列,則需要在窗體的Designer.cs文件中手動(dòng)修改與DataGridView列相關(guān)的代碼;或者你也可以通過程序動(dòng)態(tài)指定DataGridView的列。將需要顯示數(shù)據(jù)過濾的列的類型指定為DataGridViewAutoFilterTextBoxColumn類型。另外在綁定DataGridView數(shù)據(jù)源時(shí)必須使用BindingSource而不能使用如DataTable之類的普通數(shù)據(jù)源,這一點(diǎn)非常重要!在后面的代碼展示中你將會(huì)看到為什么要這么做。
這里是具體的例子:
public Form1() {InitializeComponent();// create sequence Item[] items = new Item[] { new Book{Id = 1, Price = 13.50, Genre = "Comedy", Author = "Jim Bob"}, new Book{Id = 2, Price = 8.50, Genre = "Drama", Author = "John Fox"}, new Movie{Id = 1, Price = 22.99, Genre = "Comedy", Director = "Phil Funk"},new Movie{Id = 1, Price = 13.40, Genre = "Action", Director = "Eddie Jones"}};var query = from i in itemsorderby i.Priceselect i;DataTable table = query.CopyToDataTable();BindingSource source = new BindingSource();source.DataSource = table;foreach (DataColumn col in table.Columns){DataGridViewAutoFilterTextBoxColumn commonColumn = new DataGridViewAutoFilterTextBoxColumn();commonColumn.DataPropertyName = col.ColumnName;commonColumn.HeaderText = col.ColumnName;commonColumn.Resizable = DataGridViewTriState.True;this.dataGridView1.Columns.Add(commonColumn);} this.dataGridView1.DataSource = source; }代碼中的第16行將LINQ的查詢結(jié)果轉(zhuǎn)換成了DataTable對象,相關(guān)內(nèi)容大家可以看我的另一篇文章“如何將LINQ查詢到的結(jié)果由匿名類型var轉(zhuǎn)換成DataTable對象”。另外代碼中將DataGridView的所有列的類型指定成了DataGridViewAutoFilterTextBoxColumn,使其能夠支持自定義的數(shù)據(jù)過濾功能。好了,現(xiàn)在運(yùn)行你的應(yīng)用程序,將會(huì)看到表頭上有下拉列表的小箭頭,點(diǎn)擊它并選擇下拉列表中的項(xiàng)便可實(shí)現(xiàn)DataGridView數(shù)據(jù)的排序。是不是很酷啊?不過這里還有一個(gè)小問題,那就是用戶如何知道我當(dāng)前選擇了哪個(gè)列的數(shù)據(jù)過濾,界面是不是應(yīng)該給出相應(yīng)的數(shù)據(jù)過濾信息呢?我們可以在窗體的StatusStrip控件中添加一些Label標(biāo)簽用來顯示這些信息:
1. 顯示用戶當(dāng)前選擇了多少行。這個(gè)需要將DataGridView的SelectionMode屬性設(shè)置成行選擇模式即FullRowSelect。
2. 顯示當(dāng)前DataGridView一共有多少行。
3. 顯示Filter的信息及應(yīng)用數(shù)據(jù)過濾之后的總行數(shù)。
4. 添加一個(gè)按鈕或鏈接用于移除當(dāng)前的Filter。
來看看具體的實(shí)現(xiàn)代碼及程序運(yùn)行時(shí)的效果:
private void dataGridView1_SelectionChanged(object sender, EventArgs e) {int iCount = this.dataGridView1.SelectedRows.Count;this.toolStripStatus_SelectedRows.Text = string.Format("{0} row{1} selected", iCount.ToString(), iCount > 1 ? "s" : ""); }private void dataGridView1_DataBindingComplete(object sender, DataGridViewBindingCompleteEventArgs e) {BindingSource data = this.dataGridView1.DataSource as BindingSource;if (data == null || data.DataSource == null){return;}/* Show total records number*/// Retrieve the unfiltered row count by // temporarily unfiltering the data.data.RaiseListChangedEvents = false;String oldFilter = data.Filter;data.Filter = null;int iTotalNum = data.Count;this.toolStripStatus_Total.Text = string.Format("Total of {0} record{1}.", iTotalNum.ToString(), iTotalNum > 1 ? "s" : "");data.Filter = oldFilter;data.RaiseListChangedEvents = true;/* Show filter information.*/int iFilterNum = data.Count;string filter = data.Filter;if (String.IsNullOrEmpty(filter)){this.toolStripStatus_Separator2.Visible = false;this.toolStripStatus_Filter.Visible = false;this.toolStripStatus_ShowAll.Visible = false;}else{this.toolStripStatus_Separator2.Visible = true;this.toolStripStatus_Filter.Visible = true;this.toolStripStatus_ShowAll.Visible = true;this.toolStripStatus_Filter.Text = string.Format("{0} record{1} found.", iFilterNum.ToString(), iFilterNum > 1 ? "s" : "");this.toolStripStatus_Filter.Text += " (Filter: " + filter + ")";} }private void toolStripStatus_ShowAll_Click(object sender, EventArgs e) {DataGridViewAutoFilterColumnHeaderCell.RemoveFilter(this.dataGridView1); }?
1. 當(dāng)前用戶選擇的總行數(shù)。
2. DataGridView中一共有多少行。
3. Filter的信息及使用Filter之后的數(shù)據(jù)行數(shù)。
4. 用于移除Filter的鏈接。
代碼中一共是三個(gè)事件,dataGridView1_SelectionChanged事件用于在DataGridView行被選擇時(shí)觸發(fā),用來更新StatusStrip中當(dāng)前用戶選擇的總行數(shù);dataGridView1_DataBindingComplete事件在DataGridView數(shù)據(jù)完成綁定時(shí)觸發(fā),用來更新StatusStrip中Filter的信息及使用Filter之后的數(shù)據(jù)行數(shù),以及DataGridView的數(shù)據(jù)總行數(shù),注意其中將BindingSource的RaiseListChangedEvents設(shè)置為false以取得DataGridView數(shù)據(jù)源中的真實(shí)數(shù)據(jù)行數(shù),之后再將其設(shè)置為true以獲取到Filter的相關(guān)信息;toolStripStatus_ShowAll_Click事件為用戶點(diǎn)擊Show All鏈接時(shí)觸發(fā),用于移除DataGridView中的Filter。
這里是完整的代碼:
http://pan.baidu.com/s/1eRZiw8M總結(jié)
以上是生活随笔為你收集整理的.NET WinForm程序中给DataGridView表头添加下拉列表实现数据过滤的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Jmeter录制app脚本
- 下一篇: WPF定时刷新UI界面