讓WPF中的DataGrid像Excel一樣可以篩選

来源:https://www.cnblogs.com/hsiang/archive/2023/02/28/17162319.html
-Advertisement-
Play Games

在預設情況下,WPF提供的DataGrid僅擁有數據展示等簡單功能,如果要實現像Excel一樣複雜的篩選過濾功能,則相對比較麻煩。本文以一個簡單的小例子,簡述如何通過WPF實話DataGrid的篩選功能,僅供學習分享使用,如有不足之處,還請指正。 ...


在預設情況下,WPF提供的DataGrid僅擁有數據展示等簡單功能,如果要實現像Excel一樣複雜的篩選過濾功能,則相對比較麻煩。本文以一個簡單的小例子,簡述如何通過WPF實話DataGrid的篩選功能,僅供學習分享使用,如有不足之處,還請指正。

涉及知識點

在本示例中,從數據綁定,到數據展示,涉及知識點如下所示:

  • DataGrid,要WPF提供的進行二維數據展示在列表控制項,預設功能非常簡單,但是可以通過數據模板或者控制項模板進行擴展和美化,可伸縮性很強。
  • MVVM,是Model-View-ViewModel的簡寫,主要進行數據和UI進行前後端分離,在本示例中,主要用到的MVVM第三方庫為CommunityToolkit.Mvvm,大大簡化原生MVVM的實現方式。
  • 集合視圖, 要對 DataGrid 中的數據進行分組、排序和篩選,可以將其綁定到支持這些函數的 CollectionView。 然後,可以在不影響基礎源數據的情況下處理 CollectionView 中的數據。 集合視圖中的更改反映在 DataGrid 用戶界面 (UI) 中。
  • Popup控制項,直接繼承FrameworkElement,提供了一種在單獨的視窗中顯示內容的方法,該視窗相對於指定的元素或屏幕坐標,浮動在當前Popup應用程式視窗上,可用於懸浮視窗。

示例截圖

本示例主要模仿Excel的篩選功能進行實現,右鍵標題欄打開浮動視窗,懸浮於標題欄下方,既可以通過文本框進行篩選,也可以通過篩選按鈕彈出右鍵菜單,選擇具體篩選方式,截圖如下所示:

 

選擇篩選方式,彈出視窗,如下所示:

 

 輸入篩選條件,點擊確定,或者取消篩選。如篩選學號裡面包含2的,效果如下所示:

 

 註意:以上篩選都是客戶端篩選,不會修改數據源,也不會重連資料庫。

核心源碼

在本示例中,核心源碼主要包含以下幾個部分:

前端視圖【MainWindow.xaml】源碼

主要實現了按學號,姓名,年齡三列進行篩選,既可以單列篩選,又可以組合篩選。且三列的篩選實現方式一致,僅是綁定列有差異。

  1 <Window x:Class="DemoDataGrid.MainWindow"
  2         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  3         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  4         xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  5         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  6         xmlns:local="clr-namespace:DemoDataGrid"
  7         mc:Ignorable="d"
  8         Title="DataGrid篩選示例" Height="650" Width="800">
  9     <Window.Resources>
 10         <ResourceDictionary>
 11             <CollectionViewSource x:Key="ItemsSource" Source="{Binding Path=Students}"></CollectionViewSource>
 12             <CollectionViewSource x:Key="Names" Source="{Binding Path=Names}"></CollectionViewSource>
 13             <CollectionViewSource x:Key="Nos" Source="{Binding Path=Nos}"></CollectionViewSource>
 14             <CollectionViewSource x:Key="Ages" Source="{Binding Path=Ages}"></CollectionViewSource>
 15             <Style x:Key="ListBoxStyle" TargetType="{x:Type ListBox}">
 16                 <Setter Property="ScrollViewer.CanContentScroll" Value="True"></Setter>
 17                 <Setter Property="Template">
 18                     <Setter.Value>
 19                         <ControlTemplate TargetType="{x:Type ListBox}">
 20                             <ScrollViewer x:Name="ScrollViewer" CanContentScroll="True">
 21                                 <ItemsPresenter></ItemsPresenter>
 22                             </ScrollViewer>
 23                         </ControlTemplate>
 24                     </Setter.Value>
 25                 </Setter>
 26             </Style>
 27             <Geometry x:Key="Icon_Filter">
 28                 M608 864C588.8 864 576 851.2 576 832L576 448c0-6.4 6.4-19.2 12.8-25.6L787.2 256c6.4-6.4 6.4-19.2 0-19.2 0-6.4-6.4-12.8-19.2-12.8L256 224c-12.8 0-19.2 6.4-19.2 12.8 0 6.4-6.4 12.8 6.4 19.2l198.4 166.4C441.6 428.8 448 441.6 448 448l0 256c0 19.2-12.8 32-32 32S384 723.2 384 704L384 460.8 198.4 307.2c-25.6-25.6-32-64-19.2-96C185.6 179.2 217.6 160 256 160L768 160c32 0 64 19.2 76.8 51.2 12.8 32 6.4 70.4-19.2 89.6l-192 160L633.6 832C640 851.2 627.2 864 608 864z
 29             </Geometry>
 30             <ContextMenu x:Key="queryConditionMenu" MouseLeave="ContextMenu_MouseLeave" MenuItem.Click="ContextMenu_Click">
 31                 <MenuItem FontSize="12" Header="等於" Tag="Equal"></MenuItem>
 32                 <MenuItem FontSize="12" Header="不等於"  Tag="NotEqual"></MenuItem>
 33                 <MenuItem FontSize="12" Header="開頭"  Tag="Begin"></MenuItem>
 34                 <MenuItem FontSize="12" Header="結尾"  Tag="End"></MenuItem>
 35                 <MenuItem FontSize="12" Header="包含"  Tag="In"></MenuItem>
 36                 <MenuItem FontSize="12" Header="不包含"  Tag="NotIn"></MenuItem>
 37             </ContextMenu>
 38         </ResourceDictionary>
 39         
 40     </Window.Resources>
 41     <Grid Margin="10">
 42         <Grid.RowDefinitions>
 43             <RowDefinition Height="20"></RowDefinition>
 44             <RowDefinition Height="*"></RowDefinition>
 45         </Grid.RowDefinitions>
 46         <DataGrid Grid.Row="1" x:Name="dgStudents" ItemsSource="{Binding Source={StaticResource ItemsSource} }" AutoGenerateColumns="False"
 47                               CanUserReorderColumns="True" CanUserDeleteRows="False" CanUserAddRows="False" HeadersVisibility="Column"
 48                               CanUserSortColumns="True"
 49                               VirtualizingPanel.VirtualizationMode="Recycling"
 50                               EnableColumnVirtualization="True" VirtualizingPanel.IsVirtualizingWhenGrouping="True" GridLinesVisibility="All" RowHeight="25"
 51                               SelectionUnit="FullRow" SelectionMode="Single" IsReadOnly="True" FontSize="12" 
 52                               SelectedIndex="{Binding SelectTaskItemIndex}" SelectedItem="{Binding SelectTaskItem}"
 53                               CanUserResizeColumns="True">
 54             <DataGrid.Columns>
 55                 <DataGridTextColumn Binding="{Binding Id}" Header="Id" Width="*">
 56                     
 57                 </DataGridTextColumn>
 58                 <DataGridTextColumn Binding="{Binding No}" Width="*">
 59                     <DataGridTextColumn.Header>
 60                         <TextBlock Text="學號" FontWeight="Regular" MouseRightButtonDown="TextBlock_MouseRightButtonDown" Tag="No"></TextBlock>
 61                     </DataGridTextColumn.Header>
 62                 </DataGridTextColumn>
 63                 <DataGridTextColumn Binding="{Binding Name}" Width="*">
 64                     <DataGridTextColumn.Header>
 65                         <TextBlock Text="姓名" FontWeight="Regular" MouseRightButtonDown="TextBlock_MouseRightButtonDown" Tag="Name"></TextBlock>
 66                     </DataGridTextColumn.Header>
 67                 </DataGridTextColumn>
 68                 <DataGridTextColumn Binding="{Binding Age}" Width="*">
 69                     <DataGridTextColumn.Header>
 70                         <TextBlock Text="年齡" FontWeight="Regular" MouseRightButtonDown="TextBlock_MouseRightButtonDown" Tag="Age"></TextBlock>
 71                     </DataGridTextColumn.Header>
 72                 </DataGridTextColumn>
 73             </DataGrid.Columns>
 74         </DataGrid>
 75         <Popup x:Name="popupNo" Width="135" MaxHeight="500" Height="Auto" PopupAnimation="Slide" AllowsTransparency="False" MouseLeave="popup_MouseLeave">
 76             <Border BorderBrush="LightBlue" BorderThickness="1" Padding="5" Background="AliceBlue">
 77                 <Grid>
 78                     <Grid.ColumnDefinitions>
 79                         <ColumnDefinition></ColumnDefinition>
 80                         <ColumnDefinition></ColumnDefinition>
 81                     </Grid.ColumnDefinitions>
 82                     <Grid.RowDefinitions>
 83                         <RowDefinition></RowDefinition>
 84                         <RowDefinition></RowDefinition>
 85                         <RowDefinition></RowDefinition>
 86                     </Grid.RowDefinitions>
 87                     <StackPanel Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="2" Orientation="Horizontal">
 88                         <TextBox Height="25" x:Name="txtNo" MinWidth="60" Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="2" Tag="No" TextChanged="TextBox_TextChanged"></TextBox>
 89                         <Button x:Name="btnNoFilter"  Tag="No" ClickMode="Press" Click="ButtonFilter_Click" ContextMenu="{StaticResource queryConditionMenu}">
 90                             <StackPanel Orientation="Horizontal" VerticalAlignment="Center">
 91                                 <Path Data="{StaticResource Icon_Filter}" Stroke="Gray" StrokeThickness="1" 
 92                        Height="12" Width="12" Stretch="Fill"></Path>
 93                                 <TextBlock Margin="2,0" Text="篩選" FontSize="12"></TextBlock>
 94                             </StackPanel>
 95                         </Button>
 96                     </StackPanel>
 97                     <ListBox x:Name="lbNos" ItemsSource="{Binding Source={StaticResource Nos}}" Grid.Column="0" Grid.Row="1" Grid.ColumnSpan="2" VirtualizingPanel.VirtualizationMode="Recycling" VirtualizingPanel.IsVirtualizing="True" Style="{StaticResource ListBoxStyle}">
 98                         <ListBox.ItemTemplate>
 99                             <DataTemplate>
100                                 <CheckBox Content="{Binding FilterText}" IsChecked="{Binding IsChecked}"></CheckBox>
101                             </DataTemplate>
102                         </ListBox.ItemTemplate>
103                     </ListBox>
104 
105                     <Button Tag="No" Content="取消" Width="60" HorizontalAlignment="Left" Grid.Row="2" Grid.Column="0" Click="btnCancel_Click"></Button>
106                     <Button Content="確定" Width="60" HorizontalAlignment="Right" Grid.Row="2" Grid.Column="1" Click="btnOk_Click"></Button>
107                 </Grid>
108             </Border>
109         </Popup>
110         <Popup x:Name="popupName" Width="135" MaxHeight="500" Height="Auto" PopupAnimation="Slide" AllowsTransparency="False" MouseLeave="popup_MouseLeave">
111             <Border BorderBrush="LightBlue" BorderThickness="1" Padding="5" Background="AliceBlue">
112                 <Grid>
113                     <Grid.ColumnDefinitions>
114                         <ColumnDefinition></ColumnDefinition>
115                         <ColumnDefinition></ColumnDefinition>
116                     </Grid.ColumnDefinitions>
117                     <Grid.RowDefinitions>
118                         <RowDefinition></RowDefinition>
119                         <RowDefinition></RowDefinition>
120                         <RowDefinition></RowDefinition>
121                     </Grid.RowDefinitions>
122                     <StackPanel Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="2" Orientation="Horizontal">
123                         <TextBox Height="25" x:Name="txtName" MinWidth="60" Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="2" Tag="Name" TextChanged="TextBox_TextChanged"></TextBox>
124                         <Button x:Name="btnNameFilter"  Tag="Name" Click="ButtonFilter_Click" ContextMenu="{StaticResource queryConditionMenu}">
125                             <StackPanel Orientation="Horizontal" VerticalAlignment="Center">
126                                 <Path Data="{StaticResource Icon_Filter}" Stroke="Gray" StrokeThickness="1" 
127                        Height="12" Width="12" Stretch="Fill"></Path>
128                                 <TextBlock Margin="2,0" Text="篩選" FontSize="12"></TextBlock>
129                             </StackPanel>
130                         </Button>
131                     </StackPanel>
132                     <ListBox x:Name="lbNames" ItemsSource="{Binding Source={StaticResource Names}}" Grid.Column="0" Grid.Row="1" Grid.ColumnSpan="2" VirtualizingPanel.VirtualizationMode="Recycling" VirtualizingPanel.IsVirtualizing="True" Style="{StaticResource ListBoxStyle}">
133                         <ListBox.ItemTemplate>
134                             <DataTemplate>
135                                 <CheckBox Content="{Binding FilterText}" IsChecked="{Binding IsChecked}"></CheckBox>
136                             </DataTemplate>
137                         </ListBox.ItemTemplate>
138                     </ListBox>
139 
140                     <Button Tag="No" Content="取消" Width="60" HorizontalAlignment="Left" Grid.Row="2" Grid.Column="0" Click="btnCancel_Click"></Button>
141                     <Button Content="確定" Width="60" HorizontalAlignment="Right" Grid.Row="2" Grid.Column="1" Click="btnOk_Click"></Button>
142                 </Grid>
143             </Border>
144    

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • if條件語句 if語句 if條件語法結構: if 條件語句: 滿足條件運行的代碼1 滿足條件運行的代碼2 ... ps:條件語句(可以是單個數據,即本身就是布爾類型)需返回一個布爾類型,判斷是否進入條件分支語句 if True: print('條件成⽴執⾏的代碼1') print('條件成⽴執⾏的代 ...
  • Lambda表達式 Lambda表達式理解 Lambda表達式是Jdk 8 開始新增的一種語法形式;作用:用於簡化匿名內部類的代碼寫法 註意:Lambda表達式只能簡化函數式介面的匿名內部類!!! 什麼是函數式介面? 有且僅有一個抽象方法的介面。 註意:大部分函數式介面,上面可能會有一個@Funct ...
  • 本節開始,將對 ResourceManager 中一些常見行為進行分析探究,看某些具體關鍵的行為,在 RM 中是如何流轉的。本節將深入源碼探究「啟動 ApplicationMaster」的具體流程。 ...
  • 錯誤描述 在 Spring Cloud 項目中通過 Open Feign 遠程調用時出現如下錯誤: feign.codec.EncodeException: No qualifying bean of type 'org.springframework.boot.autoconfigure.http ...
  • 之前在做某個業務中,寫了個文件傳輸的程式,程式邏輯很簡單:掃描某個目錄下的文件,對文件進行一些處理,然後把文件移動到另一個目錄。 此前在大多數運行環境里,該程式一直正常運行,直到最近在一個新環境下,出現問題:文件移動失敗。查詢日誌發現在調用file.renameTo方法返回false。我第一反應是查 ...
  • C語言對記憶體的使用劃分為以下區域: 棧區(stack)、堆區(heap)、全局區(靜態區)、常量區、代碼區。 棧區: 由編譯器自動分配釋放,按記憶體地址從高(地址)到低(地址)存儲; 棧區內容的作用域為其所定義的函數內,生命周期為函數執行期間,函數結束自動釋放; 存放局部變數、const局部變數、函數 ...
  • 在上文中分析了 HttpURLConnection的用法,功能還是比較簡單的,沒有什麼封裝 接下來看看Apache HttpClient是如何封裝httpClient的 組成 HttpClient 5 的系統架構主要由以下幾個部分組成: HttpCore:核心包,包含了 HTTP 協議的核心抽象和實 ...
  • 背景 昨天,一位朋友找到我尋求幫助。他的項目需要調用一個第三方項目的webAPI。這個webAPI本身可從header, query string中取相關信息,但同事發現他在調用時,無法按期望的那樣從query string中傳參數給到第三方webAPI (webAPI仿佛忽略了從query str ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...