棱镜/MVVM:将列绑定到DataGrid

5

我将使用一个标准的.NET DataGrid,就像这样:

<DataGrid ItemsSource="{Binding Datensaetze}" AutoGenerateColumns="False">
 <DataGrid.Columns>
   <DataGridTextColumn Header="my col 1" Binding="{Binding MyCol1}"/>
   <DataGridTextColumn Header="my col 2" Binding="{Binding MyCol2}"/>
   <DataGridTextColumn Header="my col 3" Binding="{Binding MyCol3}"/>
 </DataGrid.Columns>
</DataGrid>

这个很好地运作了。现在我想在ViewModel中定义列,而不是在xaml中设置固定列,我想动态生成它们。然而,如果我尝试将列绑定到任何东西,我会得到一个错误,说

DataGrid.Columns是只读属性,无法绑定。

有没有办法在代码后台动态绑定DataGrid列到某些东西?

1个回答

9

是的,Columns属性是只读的,因此我们无法直接绑定它。如果您想要绑定列,则可以尝试使用一个附加属性进行绑定,该附加属性会更新列。

更新
使用CollectionChanged事件中的更改差值。

public class DataGridColumnsBehavior
{
    public static readonly DependencyProperty BindableColumnsProperty =
        DependencyProperty.RegisterAttached("BindableColumns",
                                            typeof(ObservableCollection<DataGridColumn>),
                                            typeof(DataGridColumnsBehavior),
                                            new UIPropertyMetadata(null, BindableColumnsPropertyChanged));
    private static void BindableColumnsPropertyChanged(DependencyObject source, DependencyPropertyChangedEventArgs e)
    {
        DataGrid dataGrid = source as DataGrid;
        ObservableCollection<DataGridColumn> columns = e.NewValue as ObservableCollection<DataGridColumn>;
        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<DataGridColumn> value)
    {
        element.SetValue(BindableColumnsProperty, value);
    }
    public static ObservableCollection<DataGridColumn> GetBindableColumns(DependencyObject element)
    {
        return (ObservableCollection<DataGridColumn>)element.GetValue(BindableColumnsProperty);
    }
}

然后,您可以将BindableColumns属性绑定到您的ColumnsCollection。
<DataGrid AutoGenerateColumns="False"
          local:DataGridColumnsBehavior.BindableColumns="{Binding ColumnCollection}">
    <!-- ... -->
</DataGrid>

我一直在使用这个很好,但是我遇到了一个问题,我需要在后台线程上更新ObservableCollection(我正在使用Dispatcher来完成此操作)。然后,在CollectionChanged委托中,以上代码会出现错误“调用线程无法访问此对象,因为不同的线程拥有它”。有什么建议吗? - Andrew Stephens
我和安德鲁·斯蒂芬斯处于同样的境地。找不到解决办法。有人能帮忙吗? - ncourcy84
我认为在后台线程中更新绑定到UI(通过raisepropertychanged/collectionchanged)的对象之前,必须首先取消绑定项源。不能在后台线程中访问与UI绑定的任何对象。另一个选项是使用不同的对象,在后台线程中更改/访问该对象,然后使用Continue with(如果您正在使用任务),然后将该对象传递给ContinueWith块。 - Lance
这是对此解决方案的跟进问题,现在我有一个动态列,如何将动态项目源/单元格值添加到该列中? - Lance
没事了,我明白了,我刚刚创建了一个索引属性,然后使用列标题将其绑定到列。 - Lance
非常好。但是如果有ColumnCollection定义的示例代码会更好。 - Lei Yang

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接