[c#] WPF DataGrid를 다양한 수의 열에 바인딩하려면 어떻게합니까?



Answers

나는 내 연구를 계속하고 이것을하기 위해 합리적인 방법을 찾지 못했습니다. DataGrid의 Columns 속성은 실제로 바인딩 할 수있는 것이 아니며 실제로는 읽기 전용입니다.

Bryan은 AutoGenerateColumns를 사용하여 수행 할 수있는 작업을 제안했습니다. 간단한 .Net 리플렉션을 사용하여 ItemsSource에서 개체의 속성을보고 각 개체에 대한 열을 생성합니다. 아마도 각 열에 대한 속성을 사용하여 형식을 생성 할 수 있지만이 방법은 벗어나고 있습니다.

이 문제는 코드에서 너무 쉽게 소멸되므로 데이터 컨텍스트가 새 열로 업데이트 될 때마다 호출하는 간단한 확장 메서드를 사용합니다.

public static void GenerateColumns(this DataGrid dataGrid, IEnumerable<ColumnSchema> columns)
{
    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);
Question

내 WPF 응용 프로그램은 매번 다른 수의 열을 가질 수있는 데이터 집합을 생성합니다. 출력에 포함 된 것은 서식을 적용하는 데 사용될 각 열에 대한 설명입니다. 단순화 된 버전의 출력은 다음과 같습니다.

class Data
{
    IList<ColumnDescription> ColumnDescriptions { 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 파일의 데이터 바인딩으로 바꾸는 방법이 있습니까?




그리드 정의로 사용자 정의 컨트롤을 만들고 xaml에서 다양한 컬럼 정의로 '자식'컨트롤을 정의 할 수 있습니다. 부모는 열에 대한 종속성 속성과 열을로드하는 메서드가 필요합니다.

부모의:

public ObservableCollection<DataGridColumn> gridColumns
{
  get
  {
    return (ObservableCollection<DataGridColumn>)GetValue(ColumnsProperty);
  }
  set
  {
    SetValue(ColumnsProperty, value);
  }
}
public static readonly DependencyProperty ColumnsProperty =
  DependencyProperty.Register("gridColumns",
  typeof(ObservableCollection<DataGridColumn>),
  typeof(parentControl),
  new PropertyMetadata(new ObservableCollection<DataGridColumn>()));

public void LoadGrid()
{
  if (gridColumns.Count > 0)
    myGrid.Columns.Clear();

  foreach (DataGridColumn c in gridColumns)
  {
    myGrid.Columns.Add(c);
  }
}

자식 Xaml :

<local:parentControl x:Name="deGrid">           
  <local:parentControl.gridColumns>
    <toolkit:DataGridTextColumn Width="Auto" Header="1" Binding="{Binding Path=.}" />
    <toolkit:DataGridTextColumn Width="Auto" Header="2" Binding="{Binding Path=.}" />
  </local:parentControl.gridColumns>  
</local:parentControl>

마지막으로 까다로운 부분은 'LoadGrid'를 호출 할 위치를 찾는 것입니다.
나는 이것으로 고민하고 있지만 내 창 생성자에서 InitalizeComponent를 호출하여 작업 할 일이있다. (childGrid는 window.xaml의 x : name이다.)

childGrid.deGrid.LoadGrid();

관련 블로그 항목




프로그래밍 방식으로 수행하는 방법의 샘플이 있습니다.

public partial class UserControlWithComboBoxColumnDataGrid : UserControl
{
    private Dictionary<int, string> _Dictionary;
    private ObservableCollection<MyItem> _MyItems;
    public UserControlWithComboBoxColumnDataGrid() {
      _Dictionary = new Dictionary<int, string>();
      _Dictionary.Add(1,"A");
      _Dictionary.Add(2,"B");
      _MyItems = new ObservableCollection<MyItem>();
      dataGridMyItems.AutoGeneratingColumn += DataGridMyItems_AutoGeneratingColumn;
      dataGridMyItems.ItemsSource = _MyItems;

    }
private void DataGridMyItems_AutoGeneratingColumn(object sender, DataGridAutoGeneratingColumnEventArgs e)
        {
            var desc = e.PropertyDescriptor as PropertyDescriptor;
            var att = desc.Attributes[typeof(ColumnNameAttribute)] as ColumnNameAttribute;
            if (att != null)
            {
                if (att.Name == "My Combobox Item") {
                    var comboBoxColumn =  new DataGridComboBoxColumn {
                        DisplayMemberPath = "Value",
                        SelectedValuePath = "Key",
                        ItemsSource = _ApprovalTypes,
                        SelectedValueBinding =  new Binding( "Bazinga"),   
                    };
                    e.Column = comboBoxColumn;
                }

            }
        }

}
public class MyItem {
    public string Name{get;set;}
    [ColumnName("My Combobox Item")]
    public int Bazinga {get;set;}
}

  public class ColumnNameAttribute : Attribute
    {
        public string Name { get; set; }
        public ColumnNameAttribute(string name) { Name = name; }
}



필자는 다음과 같이 한 줄의 코드 만 사용하여 동적으로 열을 추가 할 수있었습니다.

MyItemsCollection.AddPropertyDescriptor(
    new DynamicPropertyDescriptor<User, int>("Age", x => x.Age));

질문에 관해서는 XAML 기반 솔루션이 아니기 때문에 DataGrid.Columns와 직접 작동하는 솔루션도 아닙니다. ITypedList를 구현하는 DataGrid 바운드 ItemsSource와 실제로 작동하며 PropertyDescriptor 검색을위한 사용자 지정 메서드를 제공합니다. 코드의 한 위치에서 그리드에 대해 "데이터 행"과 "데이터 열"을 정의 할 수 있습니다.

당신이 가지고있는 경우 :

IList<string> ColumnNames { get; set; }
//dict.key is column name, dict.value is value
Dictionary<string, string> Rows { get; set; }

예를 들면 다음과 같이 사용할 수 있습니다.

var descriptors= new List<PropertyDescriptor>();
//retrieve column name from preprepared list or retrieve from one of the items in dictionary
foreach(var columnName in ColumnNames)
    descriptors.Add(new DynamicPropertyDescriptor<Dictionary, string>(ColumnName, x => x[columnName]))
MyItemsCollection = new DynamicDataGridSource(Rows, descriptors) 

MyItemsCollection에 대한 바인딩을 사용하는 표에는 해당 열이 채워집니다. 이러한 열은 런타임에 동적으로 수정 (새로 추가되거나 기존 제거됨) 될 수 있으며 그리드는 열 컬렉션을 자동으로 새로 고칩니다.

위에서 언급 한 DynamicPropertyDescriptor는 일반 PropertyDescriptor를 업그레이드 한 것일뿐 아니라 강력한 형식의 열 정의와 몇 가지 추가 옵션을 제공합니다. DynamicDataGridSource는 기본 PropertyDescriptor를 사용하여 정밀한 이벤트를 처리합니다.




Related