How to Add Columns to a DataGrid through Binding and Map Its Cell Values

How to Add Columns to a DataGrid through Binding and Map Its Cell Values

Lance Contreras, 7 Nov 2013 CPOL

   4.94 (9 votes)

1

2

3

4

5

4.94/5 - 9 votes

μ 4.94, σa 1.04 [?]

Rate:

Add a reason or comment to your vote: x
Votes of 3 or less require a comment

How to add columns to a DataGrid through binding and map its cell values

Introduction

There are some ways on how you can dynamically bind to the columns of a DataGrid. In this article, I‘ll discuss a solution on how bind a collection to the DataGrid‘s column and display a mapped value to the cell using the row and column‘s binding.

Background

In my current project, I was tasked to create a datagrid that can have additional columns through binding. Thanks to Paul Stovell, I got an idea on how to implement a CustomBoundColumn. But I still have another problem, how can I get the cell value based on the Column and Row binding? Another problem is where should I bind the values of the new cells?

Normally, we can create a cell template and the binding of the row will apply to that template. But since we have a dynamic column, we should match the column‘s binding to the row binding in order for the data in the cell to make sense.

The main idea here is to get the Column‘s binding and the Row‘s binding then with that information, we can generate a sensible data for the cell. Then, we need a way to manage those new cell values though data binding.

Using the Code

I created a behavior for my DataGrid with the following attached properties:

  • AttachedColumnsProperty - This should be bound to an ObservableCollection of ViewModels for the additional columns.
  • MappedValuesProperty - This should be bound to an ObservableCollection of MappedValues. I created a MappedValue class that contains the binding source of the column header, the binding source of the view and the value that will be placed in the cell.
  • HeaderTemplateProperty - The column header template.
  • AttachedCellTemplateProperty - The cell template of the cell under the attached columns. This should be the template of those newly generated cells because of the attached columns.
    public class AttachedColumnBehavior
    {
        public static readonly DependencyProperty AttachedColumnsProperty =
                DependencyProperty.RegisterAttached("AttachedColumns",
                typeof(IEnumerable),
                typeof(AttachedColumnBehavior),
                new UIPropertyMetadata(null, OnAttachedColumnsPropertyChanged));

        public static readonly DependencyProperty MappedValuesProperty =
                DependencyProperty.RegisterAttached("MappedValues",
                typeof(MappedValueCollection),
                typeof(AttachedColumnBehavior),
                new UIPropertyMetadata(null, OnMappedValuesPropertyChanged));

        public static readonly DependencyProperty HeaderTemplateProperty =
                DependencyProperty.RegisterAttached("HeaderTemplate",
                typeof(DataTemplate),
                typeof(AttachedColumnBehavior),
                new UIPropertyMetadata(null, OnHeaderTemplatePropertyChanged));

        public static readonly DependencyProperty AttachedCellTemplateProperty =
                DependencyProperty.RegisterAttached("AttachedCellTemplate",
                typeof(DataTemplate),
                typeof(AttachedColumnBehavior),
                new UIPropertyMetadata(null, OnCellTemplatePropertyChanged));

        public static readonly DependencyProperty AttachedCellEditingTemplateProperty =
        DependencyProperty.RegisterAttached("AttachedCellEditingTemplate",
        typeof(DataTemplate),
        typeof(DataGrid),
        new UIPropertyMetadata(null, OnCellEditingTemplatePropertyChanged));

        private static void OnAttachedColumnsPropertyChanged
        (DependencyObject dependencyObject, DependencyPropertyChangedEventArgs e)
        {
            var dataGrid = dependencyObject as DataGrid;
            if (dataGrid == null) return;
            var columns = e.NewValue as INotifyCollectionChanged;
            if (columns != null)
            {
                columns.CollectionChanged += (sender, args) =>
                    {
                        if (args.Action == NotifyCollectionChangedAction.Remove)
                            RemoveColumns(dataGrid, args.OldItems);
                        else if(args.Action == NotifyCollectionChangedAction.Add)
                            AddColumns(dataGrid, args.NewItems);
                    };
                dataGrid.Loaded += (sender, args) => AddColumns(dataGrid, GetAttachedColumns(dataGrid));
                var items = dataGrid.ItemsSource as INotifyCollectionChanged;
                if (items != null)
                    items.CollectionChanged += (sender, args) =>
                        {
                            if (args.Action == NotifyCollectionChangedAction.Remove)
                                RemoveMappingByRow(dataGrid, args.NewItems);
                        };
            }
        }   

        private static void AddColumns(DataGrid dataGrid, IEnumerable columns)
        {
            foreach (var column in columns)
            {
                CustomBoundColumn customBoundColumn = new CustomBoundColumn()
                {
                    Header = column,
                    HeaderTemplate = GetHeaderTemplate(dataGrid),
                    CellTemplate = GetAttachedCellTemplate(dataGrid),
                    CellEditingTemplate = GetAttachedCellEditingTemplate(dataGrid),
                    MappedValueCollection = GetMappedValues(dataGrid)
                };

                dataGrid.Columns.Add(customBoundColumn);
            }
        }

        private static void RemoveColumns(DataGrid dataGrid, IEnumerable columns)
        {
            foreach (var column in columns)
            {
                DataGridColumn col = dataGrid.Columns.Where(x => x.Header == column).Single();
                GetMappedValues(dataGrid).RemoveByColumn(column);
                dataGrid.Columns.Remove(col);
            }
        }

        private static void RemoveMappingByRow(DataGrid dataGrid, IEnumerable rows)
        {
            foreach (var row in rows)
            {
                GetMappedValues(dataGrid).RemoveByRow(row);
            }
        }

        #region OnChange handlers
        private static void OnCellTemplatePropertyChanged
        (DependencyObject depObj, DependencyPropertyChangedEventArgs e)
        {

        }
        private static void OnHeaderTemplatePropertyChanged
        (DependencyObject depObj, DependencyPropertyChangedEventArgs e)
        {

        }

        private static void OnCellEditingTemplatePropertyChanged
        (DependencyObject depObj, DependencyPropertyChangedEventArgs e)
        {

        }
        private static void OnMappedValuesPropertyChanged
        (DependencyObject d, DependencyPropertyChangedEventArgs e)
        {

        }
        #endregion

        public static IEnumerable GetAttachedColumns(DependencyObject dataGrid)
        {
            return (IEnumerable)dataGrid.GetValue(AttachedColumnsProperty);
        }

        public static void SetAttachedColumns(DependencyObject dataGrid, IEnumerable value)
        {
            dataGrid.SetValue(AttachedColumnsProperty, value);
        }

        public static MappedValueCollection GetMappedValues(DependencyObject dataGrid)
        {
            return (MappedValueCollection)dataGrid.GetValue(MappedValuesProperty);
        }

        public static void SetMappedValues(DependencyObject dataGrid, MappedValueCollection value)
        {
            dataGrid.SetValue(MappedValuesProperty, value);
        }

        public static DataTemplate GetHeaderTemplate(DependencyObject dataGrid)
        {
            return (DataTemplate)dataGrid.GetValue(HeaderTemplateProperty);
        }

        public static void SetHeaderTemplate(DependencyObject dataGrid, DataTemplate value)
        {
            dataGrid.SetValue(HeaderTemplateProperty, value);
        }

        public static DataTemplate GetAttachedCellTemplate(DependencyObject dataGrid)
        {
            return (DataTemplate)dataGrid.GetValue(AttachedCellTemplateProperty);
        }

        public static void SetAttachedCellTemplate(DependencyObject dataGrid, DataTemplate value)
        {
            dataGrid.SetValue(AttachedCellTemplateProperty, value);
        }

        public static DataTemplate GetAttachedCellEditingTemplate(DependencyObject dataGrid)
        {
            return (DataTemplate)dataGrid.GetValue(AttachedCellEditingTemplateProperty);
        }

        public static void SetAttachedCellEditingTemplate(DependencyObject dataGrid, DataTemplate value)
        {
            dataGrid.SetValue(AttachedCellEditingTemplateProperty, value);
        }
    } 

Now here‘s the CustomBoundColumn, this is an extension of the DataGridTemplateColumn. Here, we will combine the binding source of the column and the row and this will be the binding of our cell template. With RowBindingand ColumnBinding, we can add MappedValue to the MappedValueCollection.

public class CustomBoundColumn : DataGridTemplateColumn//DataGridBoundColumn
{
    public DataTemplate CellTemplate { get; set; }
    public DataTemplate CellEditingTemplate { get; set; }
    public MappedValueCollection MappedValueCollection { get; set; }

    protected override FrameworkElement GenerateElement(DataGridCell cell, object dataItem)
    {
        var content = new ContentControl();
        MappedValue context = MappedValueCollection.ReturnIfExistAddIfNot(cell.Column.Header, dataItem);
        var binding = new Binding() { Source = context };
        content.ContentTemplate = cell.IsEditing ? CellEditingTemplate : CellTemplate;
        content.SetBinding(ContentControl.ContentProperty, binding);
        return content;
    }

    protected override FrameworkElement GenerateEditingElement(DataGridCell cell, object dataItem)
    {
        return GenerateElement(cell, dataItem);
    }
} 

The MappedValueCollection is just an ObservableCollection of MappedValues. I created it to easily manipulate the mapped values.

public class MappedValueCollection : ObservableCollection<MappedValue>
{
    public MappedValueCollection()
    {
    }

    public bool Exist(object ColumnBinding, object RowBinding)
    {
        return this.Count(x => x.RowBinding == RowBinding &&
        	x.ColumnBinding == ColumnBinding) > 0;
    }

    public MappedValue ReturnIfExistAddIfNot(object ColumnBinding, object RowBinding)
    {
        MappedValue value = null;

        if (Exist(ColumnBinding, RowBinding))
        {
            return this.Where(x => x.RowBinding == RowBinding &&
                                   x.ColumnBinding == ColumnBinding).Single();
        }
        else
        {
            value = new MappedValue();
            value.ColumnBinding = ColumnBinding;
            value.RowBinding = RowBinding;
            this.Add(value);
        }
        return value;
    }

    public void RemoveByColumn(object ColumnBinding)
    {
        foreach (var item in this.Where(x => x.ColumnBinding == ColumnBinding).ToList())
            this.Remove(item);
    }

    public void RemoveByRow(object RowBinding)
    {
        foreach (var item in this.Where(x => x.RowBinding == RowBinding).ToList())
            this.Remove(item);
    }
}

The MappedValue is where we bind the value of the new cells generated by newly attached columns.

public class MappedValue : ViewModelBase, IMappedValue
{
    object value;
    public object ColumnBinding { get; set; }
    public object RowBinding { get; set; }
    public object Value
    {
        get
        {
            return value;
        }
        set
        {
            if (this.value != value)
            {
                this.value = value;
                base.RaisePropertyChanged(() => Value);
            }
        }
    }
} 

In the ViewModel example, here we have three Collections, the CostingCollectionwhich will be the ItemsSourceof the DataGrid, the Suppliers which will be the additional columns to the grid and the SupplierCostValueswhich is the mapped value for the cells under the attached columns.

public class MainWindowViewModel : ViewModelBase
{
    ObservableCollection<CostViewModel> costingCollection;
    ObservableCollection<SupplierViewModel> suppliers;
    MappedValueCollection supplierCostValues;

    public MappedValueCollection SupplierCostValues
    {
        get { return supplierCostValues; }
        set
        {
            if (supplierCostValues != value)
            {
                supplierCostValues = value;
                base.RaisePropertyChanged(() => this.SupplierCostValues);
            }
        }
    }
    public ObservableCollection<SupplierViewModel> Suppliers
    {
        get { return suppliers; }
        set
        {
            if (suppliers != value)
            {
                suppliers = value;
                base.RaisePropertyChanged(() => this.Suppliers);
            }
        }
    }
    public ObservableCollection<CostViewModel> CostingCollection
    {
        get { return costingCollection; }
        set
        {
            if (costingCollection != value)
            {
                costingCollection = value;
                base.RaisePropertyChanged(() => this.CostingCollection);
            }
        }
    }

    public MainWindowViewModel()
    {
        SupplierCostValues = new MappedValueCollection();
        this.Suppliers = new ObservableCollection<SupplierViewModel>();
        this.CostingCollection = new ObservableCollection<CostViewModel>();

        this.Suppliers.Add(new SupplierViewModel(new Supplier() { SupplierId = 1,
          Currency = "PHP",
          Location = "Philippines", SupplierName = "Bench" }));
        this.Suppliers.Add(new SupplierViewModel(new Supplier() { SupplierId = 2,
          Currency = "JPY",
          Location = "Japan", SupplierName = "Uniqlo" }));
        this.Suppliers.Add(new SupplierViewModel(new Supplier() { SupplierId = 3,
          Currency = "USD",
          Location = "United States", SupplierName = "Aeropostale" }));
        this.Suppliers.Add(new SupplierViewModel(new Supplier() { SupplierId = 4,
          Currency = "HKD",
          Location = "Hong Kong", SupplierName = "Giordano" }));

        CostingCollection.Add(new CostViewModel(new Cost()
        { CostId = 1, Name = "Service Cost" }));
        CostingCollection.Add(new CostViewModel(new Cost()
        { CostId = 2, Name = "Items Cost" }));
        CostingCollection.Add(new CostViewModel(new Cost()
        { CostId = 3, Name = "Shipping Cost" }));
    }

    public ICommand RemoveCommand
    {
        get
        {
            return new RelayCommand(() =>
            {
                this.Suppliers.RemoveAt(this.Suppliers.Count - 1);
            });
        }
    }
    public ICommand AddCommand
    {
        get
        {
            return new RelayCommand(() =>
            {
                this.Suppliers.Add(new SupplierViewModel(new Supplier() { SupplierId = 1,
                  Currency = "PHP",
                  Location = "Philippines", SupplierName = "Bench" }));
            });
        }
    }
} 

The HeaderTemplate‘s binding would be a SupplierViewModel, the Row‘s binding will be CostViewModel and the CellTemplate‘s binding is a MappedValue.

<Window x:Class="BindableColumn.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:loc="clr-namespace:BindableColumn"
        xmlns:vm="clr-namespace:BindableColumn.ViewModel"
        Title="MainWindow" Height="350" Width="525">
    <Window.DataContext>
        <vm:MainWindowViewModel />
    </Window.DataContext>
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition />
            <RowDefinition />
        </Grid.RowDefinitions>
        <StackPanel Grid.Row="0">
            <StackPanel.Resources>
                <DataTemplate x:Key="headerTemplate">
                    <StackPanel>
                        <TextBlock Text="{Binding SupplierName}" />
                        <TextBlock Text="{Binding Currency}" />
                        <TextBlock Text="{Binding Location}" />
                    </StackPanel>
                </DataTemplate>
                <DataTemplate x:Key="cellTemplate">
                    <DataTemplate.Resources>
                        <loc:RowAndColumnMultiValueConverter x:Key="Converter"/>
                    </DataTemplate.Resources>
                    <StackPanel>
                        <TextBlock Text="{Binding Value}" />
                    </StackPanel>
                </DataTemplate>
                <DataTemplate x:Key="cellEditingTemplate">
                    <DataTemplate.Resources>
                        <loc:RowAndColumnMultiValueConverter x:Key="Converter" />
                    </DataTemplate.Resources>
                    <StackPanel>
                        <TextBox Text="{Binding Value}"/>
                    </StackPanel>
                </DataTemplate>
            </StackPanel.Resources>
            <DataGrid ItemsSource="{Binding CostingCollection}" x:Name="myGrid"
                      loc:AttachedColumnBehavior.HeaderTemplate="{StaticResource headerTemplate}"
                      loc:AttachedColumnBehavior.AttachedCellTemplate="{StaticResource cellTemplate}"
                      loc:AttachedColumnBehavior.AttachedColumns="{Binding Suppliers}"
                      loc:AttachedColumnBehavior.AttachedCellEditingTemplate=
                      	"{StaticResource cellEditingTemplate}"
                      loc:AttachedColumnBehavior.MappedValues="{Binding SupplierCostValues}"
                      AutoGenerateColumns="False">
                <DataGrid.Columns>
                    <DataGridTextColumn Header="CostId" Binding="{Binding CostId}"/>
                    <DataGridTextColumn Header="Name" Binding="{Binding Name}" />
                </DataGrid.Columns>
            </DataGrid>
            <Button Content="Add Column" Command="{Binding AddCommand}"/>
            <Button Content="Remove Lastcolumn" Command="{Binding RemoveCommand}" />
        </StackPanel>
    </Grid>
</Window> 

Conclusion

Using the AttachedColumns property, we can bind an observable collection, then collection changes (add/remove) will reflect to the UI. We can now add/remove a column without having to mess up with the code behind. Aside from binding to the AttachedColumns, we can always add columns via the XAML, that makes it more flexible to use.

Allowing dynamic columns gives more complexity since there will be some new cells because of the attached columns. Using the MappedValueCollection, we can play around with the values of the newly generated cells with reference to the Row and Column of the cell. Again, no code behind involved, it can be done in the ViewModel.

Practically, this is my way of implementing a dynamic DataGrid columns using the MVVM pattern. I hope this solution helps in your development.

Further Improvement

In this solution, I only used one type of view model for all the attached columns. I think we can use different type of viewmodels for each attached column by utilizing the HeaderTemplateSelector.

References

http://www.codeproject.com/Tips/676530/How-to-Add-Columns-to-a-DataGri

时间: 2024-10-09 20:32:02

How to Add Columns to a DataGrid through Binding and Map Its Cell Values的相关文章

211 Add and Search Word - Data structure design--- back tracking, map, set 待续 trie

题意: 设计个字典查询系统, 有 add 和search 两种操作, add 是加入单词到字典里, search 时 可以用 点号通配符 ".", 点号可以匹配一个字母. 分析: 当search 时为 通配符时, 如果直接用back tracking产生 a-z, 比如 有7个点号, 就得生成  26^7 个组合,会TLE. 以下是TLE的code: class WordDictionary { /** Initialize your data structure here. */ S

WPF DataGrid 之数据绑定

1. Auto generation of columns 最简单的方法莫过于让DataGrid根据数据源中的字段自动生成列了: 根据实体类的公共属性, 能够自动生成四种类型的数据列,对应关系如下: TextBox columns for string values; CheckBox columns for boolean values; ComboBox columns for enumerable values; Hyperlink columns for Uri values; 拖个Da

Jquery easy ui datagrid動態加載列問題

1.如下图效果是当选择不同的日期范围时datagrid则会加载出对应的列数 2.首先是后台组装数据,我采用的是循环并拼接DataTable数据,如下代码 //循環添加datagrid所需的表頭數據 for (int i = 0; i < table.Columns.Count; i++) { columns.AppendFormat("{{field:'{0}',title:'{1}',align:'center',width:{2}}},", table.Columns[i].

如何用easyui+JAVA 实现动态拼凑datagrid表格

先给大家看一看效果,最近一段时间都在研究这个东西. 如果我把日期间隔选宽呢?比如5月日到5月5日?下面给大家看看效果,不用担心哦 看到了吧,哈哈,这个日期都是动态生成的,下面就来跟大家分享一下这个的实现方法. 本人是用JAVA EE的后台实现的, 先来贴HTML代码: 1 <div>站点:<input class="easyui-combobox" width="200px" id="stnmCombo">  2 起始时间

Silverlight日记:动态生成DataGrid、行列装换、动态加载控件

本文主要针对使用DataGrid动态绑定数据对象,并实现行列转换效果. 一,前台绑定 <sdk:DataGrid x:Name="dataGrid2" Style="{StaticResource ResourceKey=safeDataGrid2}" /> using System; using System.Collections; using System.Collections.Generic; using System.Collections.

WPF DataGrid 示例 应用源码

界面运行效果: XMAL 源码: <Window x:Class="WpfApp1.UI.DataGridExample" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.

WPF 精修篇 DataGrid 数据源排序

原文:WPF 精修篇 DataGrid 数据源排序 效果 <DataGrid x:Name="datagrid" ItemsSource="{Binding ElementName=Mwindow, Path=Preson}" Margin="0,0,0,20"> <DataGrid.Columns> <DataGridTextColumn Binding="{Binding Name}" Hea

WPF DataGrid表头合并且动态添加列

DataGrid要实现表头合并的效果.首先使用DataGridTemplate作为列.同时修改HeaderTemplate.但是效果没有那么好且有其他问题. 真正的修改的地方是修改HeaderStyle的DataGridColumnHeader. 内容模板则是修改CellTemplate就好了 如果要是同态添加列则是需要使用继承DataGridTemplate的类. 重写GenerateElement方法,并设置好内容模板的数据源. 所以大部分的内容都是C#代码,Xaml的部分则就是datate

WPF日积月累之DataGrid样式以及操作数据模板中的控件

一.效果图 二.代码预览 1 <Window x:Class="Test.MainWindow" 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 4 xmlns:d="http://schemas.microsoft.com/exp