我的WPF应用程序生成的数据集可能每次都有不同的列数.输出中包含将用于应用格式的每列的说明.输出的简化版本可能类似于:
class Data { IListColumnDescriptions { 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文件中的数据绑定替换此代码?
这是DataGrid中绑定列的解决方法.由于Columns属性是ReadOnly,就像每个人都注意到的那样,我创建了一个名为BindableColumns的附加属性,每当集合通过CollectionChanged事件更改时,它都会更新DataGrid中的Columns.
如果我们有这个DataGridColumn的集合
public ObservableCollectionColumnCollection { 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); } }
我继续我的研究,没有找到任何合理的方法来做到这一点.DataGrid上的Columns属性不是我可以绑定的东西,实际上它是只读的.
Bryan建议可以使用AutoGenerateColumns完成某些操作,所以我看了一下.它使用简单的.Net反射来查看ItemsSource中对象的属性,并为每个对象生成一列.也许我可以动态生成一个类型,每个列都有一个属性,但这已经偏离了轨道.
由于这个问题很容易在代码中解决,我将坚持使用一个简单的扩展方法,只要用新列更新数据上下文,我就会调用它:
public static void GenerateColumns(this DataGrid dataGrid, IEnumerablecolumns) { 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);
我在Deborah Kurata的博客文章中找到了一个很好的技巧,如何在DataGrid中显示可变数量的列:
使用MVVM在Silverlight应用程序中使用动态列填充DataGrid
基本上,她创建一个DataGridTemplateColumn
并放入ItemsControl
内部,显示多个列.
我设法使用一行代码动态添加列,如下所示:
MyItemsCollection.AddPropertyDescriptor( new DynamicPropertyDescriptor("Age", x => x.Age));
关于这个问题,这不是基于XAML的解决方案(因为如前所述没有合理的方法),它既不是直接使用DataGrid.Columns运行的解决方案.它实际上与DataGrid绑定的ItemsSource一起运行,后者实现了ITypedList,因此为PropertyDescriptor检索提供了自定义方法.在代码中的一个位置,您可以为网格定义"数据行"和"数据列".
如果你有:
IListColumnNames { 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工作.