C# WPF DataGrid 数据绑定时的单元格编辑类型模板

DataGrid的DataSource绑定的ObservableCollection<UserType>属性如果在运行中会发生变动,则不能在xaml手动定义Columns,否则当该属性发生变动时,UI不会随之更新,需要使用以下方法来实现单元格编辑类型的替换:

一、

设置DataGrid的AutoGenerateColumns为True,自动生成列的同时在AutoGeneratingColumn事件的方法将e.Column属性替换为DataGridTemplateColumn实例。

xaml:

复制代码
    <DataGrid ItemsSource="{Binding DgvItemSource}" AutoGenerateColumns="True" AutoGeneratingColumn="DataGridAutoGeneratingColumn" />

cs:

复制代码
      private void DataGridAutoGeneratingColumn(object sender, DataGridAutoGeneratingColumnEventArgs e)
      {
          DataGrid dataGrid = sender as DataGrid;
          string columnName = e.PropertyName;
          DataGridTemplateColumn column = new DataGridTemplateColumn();
          e.Column = column;
          
      }

在定义的DataGridTemplateColumn实例的CellTemplate和CellEditingTemplate属性赋值新的DataTemplate实例, 创建FrameworkElementFactory实例给DataTemplate实例的VisualTree赋值;FrameworkElementFactory实例可以使用SetBinding方法绑定数据源,使用SetValue方法指定模板,模板可定义在App.xaml里,通过Application.Current.FindResource方法获取。

复制代码
 DataGridTemplateColumn column = new DataGridTemplateColumn();
 DgvCellDataTemplateSelector cellTemplate = Application.Current.FindResource("DgvCellTemplateSelector") as DgvCellDataTemplateSelector;
 DgvCellDataTemplateSelector cellEditingTemplate = Application.Current.FindResource("DgvCellEditingTemplateSelector") as DgvCellDataTemplateSelector;
 
 column.Header = columnName;

 column.CellTemplate = new DataTemplate();
 FrameworkElementFactory factory = new FrameworkElementFactory(typeof(ContentPresenter));
 factory.SetBinding(ContentPresenter.ContentProperty, new Binding(columnName));
 factory.SetValue(ContentPresenter.ContentTemplateSelectorProperty, cellTemplate);
 column.CellTemplate.VisualTree = factory;

 column.CellEditingTemplate = new DataTemplate();
 FrameworkElementFactory editingfactory = new FrameworkElementFactory(typeof(ContentPresenter));
 editingfactory.SetBinding(ContentPresenter.ContentProperty, new Binding(columnName));
 editingfactory.SetValue(ContentPresenter.ContentTemplateSelectorProperty, cellEditingTemplate);
 column.CellEditingTemplate.VisualTree = editingfactory;

二、

编写模板选择器,DgvCellDataTemplateSelector是个实现了DataTemplateSelector接口的类型:

复制代码
public class TemplateMapping : List<TemplateMappingItem>
{

}

public class TemplateMappingItem
{
    public object Value { get; set; }
    public DataTemplate Template { get; set; }
}

public class DgvCellDataTemplateSelector : DataTemplateSelector
{
    public DataTemplate DefaultTemplate { get; set; }
    public TemplateMapping TemplateMap { get; set; }

    public override DataTemplate SelectTemplate(object item, DependencyObject container)
    {
        if (item == null) return DefaultTemplate;
        if (item is DataBase data)
        {
            if (data == null) return DefaultTemplate;
            return TemplateMap?.Find(m => Equals(m.Value, data.dataType))?.Template ?? DefaultTemplate;
        }
        return DefaultTemplate;

    }
}

三、

定义模板和创建全局模板选择器实例,app.xaml:

复制代码
<Application x:Class="TestSystem.App"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:utilities="clr-namespace:TestSystem.Utilities"
             xmlns:ut="clr-namespace:TestSystem.UserType"
             StartupUri="View/MainWindow.xaml">
    <Application.Resources>
        <ResourceDictionary>

            <DataTemplate x:Key="TemplateDefault">
                <TextBlock Text="{Binding Data.Value}" />
            </DataTemplate>

            
            <utilities:TemplateMapping x:Key="DgvCellEditingDataTypeTemplateMap">
                <utilities:TemplateMappingItem 
            Value="{x:Static ut:ControlUserType+DataType.Readonly}" 
            Template="{StaticResource TemplateDefault}" />
            </utilities:TemplateMapping>
          

            <utilities:DgvCellDataTemplateSelector 
        x:Key="DgvCellEditingTemplateSelector"
        DefaultTemplate="{StaticResource TemplateDefault}"
        TemplateMap="{StaticResource DgvCellEditingDataTypeTemplateMap}" />

            

        </ResourceDictionary>
    </Application.Resources>
</Application>

四、

创建存放数据的UserType。

复制代码
  public enum DataType
  {
      Readonly
  }
  public abstract class DataBase : NotifyPropertyChanged,ICloneable
  {
      private object value;
      public DataType dataType;
      public object Value { get { return value; } set { this.value = value;OnPropertyChanged(nameof(Value)); } }

      public DataBase() { }

      public abstract object Clone();

  }
    public class ReadOnlyData : DataBase
   {
       public ReadOnlyData()
       {
           dataType = DataType.Readonly;
       }
       public override object Clone()
       {
           var clone = new ReadOnlyData
           {
               Value = this.Value
           };

           return clone;
       }
   }
相关推荐
Scout-leaf4 天前
WPF新手村教程(三)—— 路由事件
c#·wpf
用户298698530144 天前
程序员效率工具:Spire.Doc如何助你一键搞定Word表格排版
后端·c#·.net
mudtools5 天前
搭建一套.net下能落地的飞书考勤系统
后端·c#·.net
玩泥巴的5 天前
搭建一套.net下能落地的飞书考勤系统
c#·.net·二次开发·飞书
唐宋元明清21885 天前
.NET 本地Db数据库-技术方案选型
windows·c#
郑州光合科技余经理5 天前
代码展示:PHP搭建海外版外卖系统源码解析
java·开发语言·前端·后端·系统架构·uni-app·php
lindexi5 天前
dotnet DirectX 通过可等待交换链降低输入渲染延迟
c#·directx·d2d·direct2d·vortice
feifeigo1235 天前
matlab画图工具
开发语言·matlab
dustcell.5 天前
haproxy七层代理
java·开发语言·前端
norlan_jame5 天前
C-PHY与D-PHY差异
c语言·开发语言