.NET WinForm程序中给DataGridView表头添加下拉列表实现数据过滤

简介:

我们见过Excel中的数据过滤功能,可以通过点击表头上的下拉列表来实现数据的过滤,这个功能很实用,省去了我们需要在程序中单独设计数据的查询过滤模块,功能直接依赖于数据绑定控件DataGridView。先来看看Excel中的数据过滤功能。

  要想在DataGridView中实现类似于Excel的这种功能其实也并非难事。来看看msdn上的一篇文章,上面有详细的介绍,不过目前只有全英文的版本。http://msdn.microsoft.com/en-us/library/aa480727.aspx。里面提供的下载示例我这里也可以提供一份:DataGridViewAutoFilter.zip

  文章讲了很多有关如何实现数据过滤的知识,如果你有耐心可以通读一遍,应该能有不小的收获。其实这里面的原理就是我们需要自定义一种DataGridViewColumn,它能支持用户通过点击表头上的下拉列表来实现DataGridView的数据过滤。自定义的DataGridViewColumn可以继承自现有的DataGridViewTextBoxColumn类型,另外还需要自定义一个继承自DataGridViewColumnHeaderCell的类型,它负责在DataGridView表头上呈现一个下拉列表,并完成数据过滤的选择功能。下载上面的DataGridViewAutoFilter.zip压缩包,将里面对应编程语言中的DataGridViewAutoFilterColumnHeaderCell.cs和DataGridAutoFilterTextBoxColumn.cs两个文件加入到你的工程中。然后需要重新定义DataGridView中的列,如果你是手动指定DataGridView的列,则需要在窗体的Designer.cs文件中手动修改与DataGridView列相关的代码;或者你也可以通过程序动态指定DataGridView的列。将需要显示数据过滤的列的类型指定为DataGridViewAutoFilterTextBoxColumn类型。另外在绑定DataGridView数据源时必须使用BindingSource而不能使用如DataTable之类的普通数据源,这一点非常重要!在后面的代码展示中你将会看到为什么要这么做。

  这里是具体的例子:

复制代码
 1  public  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 " }, 
 7                                   new  Book{Id  =   2 , Price  =   8.50 , Genre  =   " Drama " , Author  =   " John Fox " },  
 8                                   new  Movie{Id  =   1 , Price  =   22.99 , Genre  =   " Comedy " , Director  =   " Phil Funk " },
 9                                   new  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 
20       foreach  (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;
26           this .dataGridView1.Columns.Add(commonColumn);
27      }            
28 
29       this .dataGridView1.DataSource  =  source;
30  }
复制代码

  代码中的第16行将LINQ的查询结果转换成了DataTable对象,相关内容大家可以看我的另一篇文章“如何将LINQ查询到的结果由匿名类型var转换成DataTable对象”。另外代码中将DataGridView的所有列的类型指定成了DataGridViewAutoFilterTextBoxColumn,使其能够支持自定义的数据过滤功能。好了,现在运行你的应用程序,将会看到表头上有下拉列表的小箭头,点击它并选择下拉列表中的项便可实现DataGridView数据的排序。是不是很酷啊?不过这里还有一个小问题,那就是用户如何知道我当前选择了哪个列的数据过滤,界面是不是应该给出相应的数据过滤信息呢?我们可以在窗体的StatusStrip控件中添加一些Label标签用来显示这些信息:

  1. 显示用户当前选择了多少行。这个需要将DataGridView的SelectionMode属性设置成行选择模式即FullRowSelect。

  2. 显示当前DataGridView一共有多少行。

  3. 显示Filter的信息及应用数据过滤之后的总行数。

  4. 添加一个按钮或链接用于移除当前的Filter。

  来看看具体的实现代码及程序运行时的效果:

复制代码
 1  private   void  dataGridView1_SelectionChanged( object  sender, EventArgs e)
 2  {
 3       int  iCount  =   this .dataGridView1.SelectedRows.Count;
 4       this .toolStripStatus_SelectedRows.Text  =   string .Format( " {0} row{1} selected " , iCount.ToString(), iCount  >   1   ?   " s "  :  "" );
 5  }
 6 
 7  private   void  dataGridView1_DataBindingComplete( object  sender, DataGridViewBindingCompleteEventArgs e)
 8  {
 9      BindingSource data  =   this .dataGridView1.DataSource  as  BindingSource;
10       if  (data  ==   null   ||  data.DataSource  ==   null )
11      {
12           return ;
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 ;
21       int  iTotalNum  =  data.Count;
22       this .toolStripStatus_Total.Text  =   string .Format( " Total of {0} record{1}. " , iTotalNum.ToString(), iTotalNum  >   1   ?   " s "  :  "" );
23      data.Filter  =  oldFilter;
24      data.RaiseListChangedEvents  =   true ;
25 
26       /*  Show filter information. */
27       int  iFilterNum  =  data.Count;
28       string  filter  =  data.Filter;
29       if  (String.IsNullOrEmpty(filter))
30      {
31           this .toolStripStatus_Separator2.Visible  =   false ;
32           this .toolStripStatus_Filter.Visible  =   false ;
33           this .toolStripStatus_ShowAll.Visible  =   false ;
34      }
35       else
36      {
37           this .toolStripStatus_Separator2.Visible  =   true ;
38           this .toolStripStatus_Filter.Visible  =   true ;
39           this .toolStripStatus_ShowAll.Visible  =   true ;
40           this .toolStripStatus_Filter.Text  =   string .Format( " {0} record{1} found. " , iFilterNum.ToString(), iFilterNum  >   1   ?   " s "  :  "" );
41           this .toolStripStatus_Filter.Text  +=   "  (Filter:  "   +  filter  +   " ) " ;
42      }
43  }
44 
45  private   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。

  这里是完整的代码:WindowsFormsApplication2.zip

本文转自Jaxu博客园博客,原文链接:http://www.cnblogs.com/jaxu/archive/2011/08/04/2127365.html,如需转载请自行联系原作者



相关文章
|
14天前
|
算法 Java 测试技术
Benchmark.NET:让 C# 测试程序性能变得既酷又简单
Benchmark.NET是一款专为 .NET 平台设计的性能基准测试框架,它可以帮助你测量代码的执行时间、内存使用情况等性能指标。它就像是你代码的 "健身教练",帮助你找到瓶颈,优化性能,让你的应用跑得更快、更稳!希望这个小教程能让你在追求高性能的路上越走越远,享受编程带来的无限乐趣!
61 13
|
26天前
|
前端开发 Android开发
WinForm 直接运行 Admin.NET
本文介绍了如何将 Admin.NET 以 WinForm 桌面程序模式运行,简化了手动配置 Web 服务的过程,便于演示和作为单机软件使用。通过添加特定 NuGet 包、修改 `Program.cs` 和 `Form1.cs` 文件,并调整项目配置,最终实现了在 WinForm 中嵌入 WebView 组件显示 Admin.NET 界面的效果。
22 0
WinForm 直接运行 Admin.NET
|
2月前
|
SQL XML 关系型数据库
入门指南:利用NHibernate简化.NET应用程序的数据访问
【10月更文挑战第13天】NHibernate是一个面向.NET的开源对象关系映射(ORM)工具,它提供了从数据库表到应用程序中的对象之间的映射。通过使用NHibernate,开发者可以专注于业务逻辑和领域模型的设计,而无需直接编写复杂的SQL语句来处理数据持久化问题。NHibernate支持多种数据库,并且具有高度的灵活性和可扩展性。
44 2
winform .net6 和 framework 的图表控件,为啥项目中不存在chart控件,该如何解决?
本文讨论了在基于.NET 6和.NET Framework的WinForms项目中添加图表控件的不同方法。由于.NET 6的WinForms项目默认不包含Chart控件,可以通过NuGet包管理器安装如ScottPlot等图表插件。而对于基于.NET Framework的WinForms项目,Chart控件是默认存在的,也可以通过NuGet安装额外的图表插件,例如LiveCharts。文中提供了通过NuGet添加图表控件的步骤和截图说明。
winform .net6 和 framework 的图表控件,为啥项目中不存在chart控件,该如何解决?
|
2月前
|
XML 存储 安全
C#开发的程序如何良好的防止反编译被破解?ConfuserEx .NET混淆工具使用介绍
C#开发的程序如何良好的防止反编译被破解?ConfuserEx .NET混淆工具使用介绍
96 0
|
3月前
|
Ubuntu 持续交付 API
如何使用 dotnet pack 打包 .NET 跨平台程序集?
`dotnet pack` 是 .NET Core 的 NuGet 包打包工具,用于将代码打包成 NuGet 包。通过命令 `dotnet pack` 可生成 `.nupkg` 文件。使用 `--include-symbols` 和 `--include-source` 选项可分别创建包含调试符号和源文件的包。默认情况下,`dotnet pack` 会先构建项目,可通过 `--no-build` 跳过构建。此外,还可以使用 `--output` 指定输出目录、`-c` 设置配置等。示例展示了创建类库项目并打包的过程。更多详情及命令选项,请参考官方文档。
234 11
|
3月前
|
存储 运维
.NET开发必备技巧:使用Visual Studio分析.NET Dump,快速查找程序内存泄漏问题!
.NET开发必备技巧:使用Visual Studio分析.NET Dump,快速查找程序内存泄漏问题!
|
3月前
|
自然语言处理 C# 图形学
使用dnSpyEx对.NET Core程序集进行反编译、编辑和调试
使用dnSpyEx对.NET Core程序集进行反编译、编辑和调试
|
3月前
|
开发框架 前端开发 JavaScript
ASP.NET MVC 教程
ASP.NET 是一个使用 HTML、CSS、JavaScript 和服务器脚本创建网页和网站的开发框架。
48 7