WPF MvvM DataGrid动态列

2Fa*_*uBR 18 wpf datagrid mvvm

我正在搜索如何以MvvM方式从ToolKit动态创建DataGrid的列.但看起来这是不可能的!

是否有一些必须做同样的事情?

没有必要创建一个来自DataGrid的usercontrol或其他控件,我只想将网格的de ItemSource设置为我的自定义对象,在某些时候我想在运行时基于类型定义网格的列对象.

那可能吗 ?

干杯

ctr*_*373 10

我要说这可能不是最好的解决方案,可能在某些情况下不起作用,但你可以尝试一下,看看它是否适用于你想要的东西.我只是鞭打了它,所以它可能有一些错误.它仍然会涉及一些代码,但它会让你的模型不知道你的视图.

您需要做的是创建一个扩展属性,允许您绑定DataGrid上的Columns属性.这是我放在一起的一个例子.

using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.Windows;
using System.Windows.Controls;

public static class DataGridExtension
{
    public static ObservableCollection<DataGridColumn> GetColumns(DependencyObject obj)
    {
        return (ObservableCollection<DataGridColumn>)obj.GetValue(ColumnsProperty);
    }

    public static void SetColumns(DependencyObject obj, ObservableCollection<DataGridColumn> value)
    {
        obj.SetValue(ColumnsProperty, value);
    }

    public static readonly DependencyProperty ColumnsProperty =  
           DependencyProperty.RegisterAttached("Columns",
           typeof(ObservableCollection<DataGridColumn>),
           typeof(DataGridExtension),
           new UIPropertyMetadata (new ObservableCollection<DataGridColumn>(),
           OnDataGridColumnsPropertyChanged));


    private static void OnDataGridColumnsPropertyChanged(
           DependencyObject d,
           DependencyPropertyChangedEventArgs e)
    {
        if (d.GetType() == typeof(DataGrid))
        {
            DataGrid myGrid = d as DataGrid;

            ObservableCollection<DataGridColumn> Columns = 
                 (ObservableCollection<DataGridColumn>)e.NewValue;

            if(Columns != null)
            {
                myGrid.Columns.Clear();

                if (Columns != null && Columns.Count > 0)
                {
                    foreach (DataGridColumn dataGridColumn in Columns)
                    {
                        myGrid.Columns.Add(dataGridColumn);
                    }
                }


                Columns.CollectionChanged += delegate(object sender,
                                 NotifyCollectionChangedEventArgs args)
                     {
                       if(args.NewItems != null)
                       {
                         foreach (DataGridColumn column 
                              in args.NewItems.Cast<DataGridColumn>())
                         {
                             myGrid.Columns.Add(column);
                         }
                       }

                       if(args.OldItems != null)
                       {
                         foreach (DataGridColumn column 
                                 in args.OldItems.Cast<DataGridColumn>())
                         {
                           myGrid.Columns.Remove(column);
                         }
                       }
                    };
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

然后将它附加到DataGrid,就像这样(其中columns是视图模型上的ObservableCollection属性)

<Controls:DataGrid AutoGenerateColumns="False" 
            DataGridExtension.Columns="{Binding Columns}" />
Run Code Online (Sandbox Code Playgroud)

如果你开始添加和删除列,我不确定它会如何响应,但它似乎可以从我的基本测试开始.祝好运!

  • 上面的代码通常可以完成工作。但是,当您隐藏数据网格并使其再次可见时,它将失败。因为columns属性将显示隐藏时有0列,并且columns属性被更改,所以将触发回调并尝试再次添加列,但实际上列确实存在于数据网格中,因此代码将失败,并指出无法添加重复的列。 (2认同)