当前位置:  开发笔记 > 编程语言 > 正文

如何将WPF DataGrid绑定到可变数量的列?

如何解决《如何将WPFDataGrid绑定到可变数量的列?》经验,为你挑选了4个好方法。

我的WPF应用程序生成的数据集可能每次都有不同的列数.输出中包含将用于应用格式的每列的说明.输出的简化版本可能类似于:

class Data
{
    IList ColumnDescriptions { get; set; }
    string[][] Rows { get; set; }
}

此类在WPF DataGrid上设置为DataContext,但实际上我以编程方式创建列:

for (int i = 0; i < data.ColumnDescriptions.Count; i++)
{
    dataGrid.Columns.Add(new DataGridTextColumn
    {
        Header = data.ColumnDescriptions[i].Name,
        Binding = new Binding(string.Format("[{0}]", i))
    });
}

有没有办法用XAML文件中的数据绑定替换此代码?



1> Fredrik Hedb..:

这是DataGrid中绑定列的解决方法.由于Columns属性是ReadOnly,就像每个人都注意到的那样,我创建了一个名为BindableColumns的附加属性,每当集合通过CollectionChanged事件更改时,它都会更新DataGrid中的Columns.

如果我们有这个DataGridColumn的集合

public ObservableCollection ColumnCollection
{
    get;
    private set;
}

然后我们可以像这样将BindableColumns绑定到ColumnCollection


附加属性BindableColumns

public class DataGridColumnsBehavior
{
    public static readonly DependencyProperty BindableColumnsProperty =
        DependencyProperty.RegisterAttached("BindableColumns",
                                            typeof(ObservableCollection),
                                            typeof(DataGridColumnsBehavior),
                                            new UIPropertyMetadata(null, BindableColumnsPropertyChanged));
    private static void BindableColumnsPropertyChanged(DependencyObject source, DependencyPropertyChangedEventArgs e)
    {
        DataGrid dataGrid = source as DataGrid;
        ObservableCollection columns = e.NewValue as ObservableCollection;
        dataGrid.Columns.Clear();
        if (columns == null)
        {
            return;
        }
        foreach (DataGridColumn column in columns)
        {
            dataGrid.Columns.Add(column);
        }
        columns.CollectionChanged += (sender, e2) =>
        {
            NotifyCollectionChangedEventArgs ne = e2 as NotifyCollectionChangedEventArgs;
            if (ne.Action == NotifyCollectionChangedAction.Reset)
            {
                dataGrid.Columns.Clear();
                foreach (DataGridColumn column in ne.NewItems)
                {
                    dataGrid.Columns.Add(column);
                }
            }
            else if (ne.Action == NotifyCollectionChangedAction.Add)
            {
                foreach (DataGridColumn column in ne.NewItems)
                {
                    dataGrid.Columns.Add(column);
                }
            }
            else if (ne.Action == NotifyCollectionChangedAction.Move)
            {
                dataGrid.Columns.Move(ne.OldStartingIndex, ne.NewStartingIndex);
            }
            else if (ne.Action == NotifyCollectionChangedAction.Remove)
            {
                foreach (DataGridColumn column in ne.OldItems)
                {
                    dataGrid.Columns.Remove(column);
                }
            }
            else if (ne.Action == NotifyCollectionChangedAction.Replace)
            {
                dataGrid.Columns[ne.NewStartingIndex] = ne.NewItems[0] as DataGridColumn;
            }
        };
    }
    public static void SetBindableColumns(DependencyObject element, ObservableCollection value)
    {
        element.SetValue(BindableColumnsProperty, value);
    }
    public static ObservableCollection GetBindableColumns(DependencyObject element)
    {
        return (ObservableCollection)element.GetValue(BindableColumnsProperty);
    }
}


这不是解决方案.主要原因是您在ViewModel中使用UI类.当您尝试创建一些页面切换时它也不起作用.当切换回具有这样的数据网格的页面时,你将得到一行"dataGrid.Columns.Add(column)".带有标题'X'的DataGridColumn已经存在于DataGrid的Columns集合中.DataGrids不能共享列,也不能包含重复的列实例.
您使用columns集合的`CollectionChanged`事件注册事件处理程序,但是您永远不会注销它.这样,只要视图模型存在,`DataGrid`就会保持活动状态,即使同时替换了首先包含`DataGrid`的控件模板也是如此.当不再需要`DataGrid`时,有没有保证再次注册该事件处理程序的方法?
完美的解决方案!可能你需要在BindableColumnsPropertyChanged中做一些其他的事情:1.在访问它之前检查dataGrid是否为null并抛出一个异常,其中有关于仅绑定到DataGrid的良好解释.2.检查e.OldValue是否为null并取消订阅CollectionChanged事件以防止内存泄漏.只是为了你的说服力.

2> Generic Erro..:

我继续我的研究,没有找到任何合理的方法来做到这一点.DataGrid上的Columns属性不是我可以绑定的东西,实际上它是只读的.

Bryan建议可以使用AutoGenerateColumns完成某些操作,所以我看了一下.它使用简单的.Net反射来查看ItemsSource中对象的属性,并为每个对象生成一列.也许我可以动态生成一个类型,每个列都有一个属性,但这已经偏离了轨道.

由于这个问题很容易在代码中解决,我将坚持使用一个简单的扩展方法,只要用新列更新数据上下文,我就会调用它:

public static void GenerateColumns(this DataGrid dataGrid, IEnumerable columns)
{
    dataGrid.Columns.Clear();

    int index = 0;
    foreach (var column in columns)
    {
        dataGrid.Columns.Add(new DataGridTextColumn
        {
            Header = column.Name,
            Binding = new Binding(string.Format("[{0}]", index++))
        });
    }
}

// E.g. myGrid.GenerateColumns(schema);


不,它不会.不管怎样都不提供链接,因为该解决方案的结果完全不同!
似乎Mealek的解决方案更加通用,并且在直接使用C#代码有问题的情况下很有用,例如在ControlTemplates中.
这是链接:http://blogs.msmvps.com/deborahk/populating-a-datagrid-with-dynamic-columns-in-a-silverlight-application-using-mvvm/

3> Lukas Cenovs..:

我在Deborah Kurata的博客文章中找到了一个很好的技巧,如何在DataGrid中显示可变数量的列:

使用MVVM在Silverlight应用程序中使用动态列填充DataGrid

基本上,她创建一个DataGridTemplateColumn并放入ItemsControl内部,显示多个列.



4> doblak..:

我设法使用一行代码动态添加列,如下所示:

MyItemsCollection.AddPropertyDescriptor(
    new DynamicPropertyDescriptor("Age", x => x.Age));

关于这个问题,这不是基于XAML的解决方案(因为如前所述没有合理的方法),它既不是直接使用DataGrid.Columns运行的解决方案.它实际上与DataGrid绑定的ItemsSource一起运行,后者实现了ITypedList,因此为PropertyDescriptor检索提供了自定义方法.在代码中的一个位置,您可以为网格定义"数据行"和"数据列".

如果你有:

IList ColumnNames { get; set; }
//dict.key is column name, dict.value is value
Dictionary Rows { get; set; }

你可以使用例如:

var descriptors= new List();
//retrieve column name from preprepared list or retrieve from one of the items in dictionary
foreach(var columnName in ColumnNames)
    descriptors.Add(new DynamicPropertyDescriptor(ColumnName, x => x[columnName]))
MyItemsCollection = new DynamicDataGridSource(Rows, descriptors) 

并且使用绑定到MyItemsCollection的网格将填充相应的列.这些列可以在运行时动态修改(新添加或现有删除),网格将自动刷新它的列集合.

上面提到的DynamicPropertyDescriptor只是对常规PropertyDescriptor的升级,并提供了强类型列定义以及一些其他选项.否则,DynamicDataGridSource将使用基本的PropertyDescriptor工作.

推荐阅读
LEEstarmmmmm
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有