.NETWinForm程序中给DataGridView表头添加下拉列表实现数
据过滤
  我们见过Excel中的数据过滤功能,可以通过点击表头上的下拉列表来实现数据的过滤,这个功能很实⽤,省去了我们需要在程序中单独设计数据的查询过滤模块,功能直接依赖于数据绑定控件DataGridView。先来看看Excel中的数据过滤功能。
  要想在DataGridView中实现类似于Excel的这种功能其实也并⾮难事。来看看msdn上的⼀篇⽂章,上⾯有详细的介绍,不过⽬前只有全英⽂的版本。。⾥⾯提供的下载⽰例我这⾥也可以提供⼀份:
  ⽂章讲了很多有关如何实现数据过滤的知识,如果你有耐⼼可以通读⼀遍,应该能有不⼩的收获。其实这⾥⾯的原理就是我们需要⾃定义⼀种DataGridViewColumn,它能⽀持⽤户通过点击表头上的下拉列表来实现DataGridView的数据过滤。⾃定义的DataGridViewColumn 可以继承⾃现有的DataGridViewTextBoxColumn类型,另外还需要⾃定义⼀个继承⾃DataGridViewColumnHeaderCell的类型,它负责在DataGridView表头上呈现⼀个下拉列表,并完成数据过滤的选择功能。下载上⾯的DataGridViewAutoFilter.zip压缩包,将⾥⾯对应编程语⾔中的DataGridViewAutoFilterColumnHeaderCell.cs和DataGridAutoFilterTextBoxColumn.cs两个⽂件加⼊到你的⼯程中。然后需要重新定义DataGridView中的列,如果你是⼿动指定DataGridView的列,则需要在窗体的Designer.cs⽂件中⼿动修改与DataGridView列相关的代码;或者你也可以通过程序动态指定DataGridView的列。将需要显⽰数据过滤的列的类型指定为DataGridViewAutoFilterTextBoxColumn类型。另外在绑定DataGridView数据源时必须使⽤BindingSource⽽不能使⽤如DataTable之类的普通数据源,这⼀点⾮常重要!在后⾯的代码展⽰中你将会看到为什么要这么做。
  这⾥是具体的例⼦:
1public Form1()
2 {
3    InitializeComponent();
4
5// create sequence
6    Item[] items = new Item[] { new Book{Id = 1, Price = 13.50, Genre = "Comedy", Author = "Jim Bob"},
7new Book{Id = 2, Price = 8.50, Genre = "Drama", Author = "John Fox"},
8new Movie{Id = 1, Price = 22.99, Genre = "Comedy", Director = "Phil Funk"},
9new Movie{Id = 1, Price = 13.40, Genre = "Action", Director = "Eddie Jones"}};
10
11
12    var query = from i in items
13                orderby i.Price
14                select i;
15
16    DataTable table = query.CopyToDataTable();
17    BindingSource source = new BindingSource();
18    source.DataSource = table;
19
20foreach (DataColumn col in table.Columns)
21    {
22        DataGridViewAutoFilterTextBoxColumn commonColumn = new DataGridViewAutoFilterTextBoxColumn();
23        commonColumn.DataPropertyName = col.ColumnName;
24        commonColumn.HeaderText = col.ColumnName;
25        commonColumn.Resizable = DataGridViewTriState.True;
26this.dataGridView1.Columns.Add(commonColumn);
27    }
28
datagridview数据源
29this.dataGridView1.DataSource = source;
30 }
  代码中的第16⾏将LINQ的查询结果转换成了DataTable对象,相关内容⼤家可以看我的另⼀篇⽂章“”。另外代码中将DataGridView的所有列的类型指定成了DataGridViewAutoFilterTextBoxColumn,使其能够⽀持⾃定义的数据过滤功能。好了,现在运⾏你的应⽤程序,将会看到表头上有下拉列表的⼩箭头,点击它并选择下拉列表中的项便可实现DataGridView数据的排序。是不是很酷啊?不过这⾥还有⼀个⼩问题,那就是⽤户如何知道我当前选择了哪个列的数据过滤,界⾯是不是应该给出相应的数据过滤信息呢?我们可以在窗体的StatusStrip控件中添加⼀些Label标签⽤来显⽰这些信息:
  1. 显⽰⽤户当前选择了多少⾏。这个需要将DataGridView的SelectionMode属性设置成⾏选择模式即FullRowSelect。
  2. 显⽰当前DataGridView⼀共有多少⾏。
  3. 显⽰Filter的信息及应⽤数据过滤之后的总⾏数。
  4. 添加⼀个按钮或链接⽤于移除当前的Filter。
  来看看具体的实现代码及程序运⾏时的效果:
1private void dataGridView1_SelectionChanged(object sender, EventArgs e)
2 {
3int iCount = this.dataGridView1.SelectedRows.Count;
5 }
6
7private void dataGridView1_DataBindingComplete(object sender, DataGridViewBindingCompleteEventArgs e)
8 {
9    BindingSource data = this.dataGridView1.DataSource as BindingSource;
10if (data == null || data.DataSource == null)
11    {
12return;
13    }
14
15/* Show total records number*/
16// Retrieve the unfiltered row count by
17    // temporarily unfiltering the data.
18    data.RaiseListChangedEvents = false;
19    String oldFilter = data.Filter;
20    data.Filter = null;
21int iTotalNum = data.Count;
23    data.Filter = oldFilter;
24    data.RaiseListChangedEvents = true;
25
26/* Show filter information.*/
27int iFilterNum = data.Count;
28string filter = data.Filter;
29if (String.IsNullOrEmpty(filter))
30    {
34    }
35else
36    {
42    }
43 }
44
45private void toolStripStatus_ShowAll_Click(object sender, EventArgs e)
46 {
47    DataGridViewAutoFilterColumnHeaderCell.RemoveFilter(this.dataGridView1);
48 }
  1. 当前⽤户选择的总⾏数。
  2. DataGridView中⼀共有多少⾏。
  3. Filter的信息及使⽤Filter之后的数据⾏数。
  4. ⽤于移除Filter的链接。
  代码中⼀共是三个事件,dataGridView1_SelectionChanged事件⽤于在DataGridView⾏被选择时触发,⽤来更新StatusStrip中当前⽤户选择的总⾏数;dataGridView1_DataBindingComplete事件在DataGridView数据完成绑定时触发,⽤来更新StatusStrip中Filter的信息及使⽤Filter之后的数据⾏数,以及DataGridView的数据总⾏数,注意其中将BindingSource的RaiseListChangedEvents设置为false以取得DataGridView数据源中的真实数据⾏数,之后再将其设置为true以获取到Filter的相关信息;toolStripStatus_ShowAll_Click事件为⽤户点击Show All链接时触发,⽤于移除DataGridView中的Filter。
  这⾥是完整的代码: