0

我正在尝试为 WPF 中的 DataGrid 创建一个过滤系统,就像在链接中一样 -

http://www.codeproject.com/Articles/42227/Automatic-WPF-Toolkit-DataGrid-Filtering

在此处输入图像描述

我正在使用他们正在使用的同一个库,但问题是我需要将我的 DataGrid 绑定到数据表......

并且有错误的地方..只要ItemSource是LIST,过滤器库就可以完美地工作,但是当ItemSource是DataTable时停止工作......

任何替代方案或建议???工作示例赞赏..

我正在使用 AutoColumnGeneration=True 因为我不知道需要填充多少列

4

1 回答 1

5

表中的每个 DataRow 都包含一个平面对象数组,其类型只能通过从 DataColumns 数组中提取来访问。由于 Array 中的所有内容都已装箱,因此依赖于强类型集合的库(如 List of T)将不兼容。这应该可以解释为什么它不起作用。

你写道这很紧急,我不得不在一个项目中做同样类型的事情,而且导火索很短,我将概述我的方法。此答案仅适用于紧急情况。

这是一个视图模型...

public class ViewModel : INotifyPropertyChanged
{
    public CollectionView MyCollectionView { get; set; }
    public ViewModel(DataTable dataTable)
    {
        MyCollectionView = CollectionViewSource.GetDefaultView(dataTable.Rows) as CollectionView;
        if (MyCollectionView != null)
        {
            MyCollectionView.Filter = o => (o as DataRow).ItemArray[0].ToString().Contains("2");
        }
    }
    public event PropertyChangedEventHandler PropertyChanged;
    private void OnPropertyChanged(string name)
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(name));
        }
    }
}

构造函数采用 DataTable 并使用 Rows 作为 CollectionView 的绑定源。由于 CollectionView 具有内置的过滤和排序属性,因此它是用作绑定源的绝佳工具。同样如图所示,构造函数为示例添加了一个过滤器以帮助您入门。

要将数据呈现给用户,请考虑以下 Xaml...

        <DataGrid ItemsSource="{Binding MyCollectionView}"
                  AutoGenerateColumns="False"
                  CanUserSortColumns="True"
                  IsReadOnly="True"
                  >
            <DataGrid.Columns>
                <DataGridTextColumn Header="Name"    Binding="{Binding ., Converter=
                       {db:DbConverter}, ConverterParameter=PersonName}"/>
                <DataGridTextColumn Header="Address"    Binding="{Binding ., Converter=
                        {db:DbConverter}, ConverterParameter=PersonAddress}"/>
            </DataGrid.Columns>
        </DataGrid>

这是一个 DataGrid,其 ItemsSource 绑定到在 ViewModel 中声明和分配的 CollectionView。每列数据与一个参数一起通过转换器传递,该参数告诉转换器它需要做什么。

为了限制一切,这里是转换器......

public class DbConverter : MarkupExtension, IValueConverter
{
    private static readonly Dictionary<object, int> ParameterToColumnMapping;
    static DbConverter()
    {
        ParameterToColumnMapping = new Dictionary<object, int>
            {
                {"PersonName", 0}, {"PersonAddress", 1}
            };
    }
    public object Convert(object value, Type targetType, object parameter, 
                          System.Globalization.CultureInfo culture)
    {
        DataRow dr = value as DataRow;
        if (dr != null)
        {
            if (ParameterToColumnMapping.ContainsKey(parameter))
            {
                return dr.ItemArray[ParameterToColumnMapping[parameter]];
            }
        }
        return value;
    }
    public object ConvertBack(object value, Type targetType, object parameter, 
               System.Globalization.CultureInfo culture)
    {
        return null;
    }
    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        return this;
    }
}

当您有 DataTable 和紧急情况时,这应该让您回到正轨进行过滤和呈现。全部编码大约需要一个小时。另请注意,我的绑定到Rows ,但如果需要,您也可以绑定到DefaultView 。

于 2013-06-25T15:11:00.660 回答