我有一个ObservableCollection<Dictionary>
,想要将其绑定到DataGrid
。
ObservableDictionary<String,Object> NewRecord1 = new ObservableDictionary<string,object>();
Dictionary<String,Object> Record1 = new Dictionary<string,object>();
Record1.Add("FirstName", "FName1");
Record1.Add("LastName", "LName1");
Record1.Add("Age", "32");
DictRecords.Add(Record1);
Dictionary<String, Object> Record2 = new Dictionary<string, object>();
NewRecord2.Add("FirstName", "FName2");
NewRecord2.Add("LastName", "LName2");
NewRecord2.Add("Age", "42");
DictRecords.Add(Record2);
我希望密钥成为DataGrid
的标头,每个Dictionary
项的值都是行。设置ItemsSource
不起作用。
答案 0 :(得分:24)
您可以使用可绑定的动态字典。这会将每个字典条目公开为属性。
/// <summary>
/// Bindable dynamic dictionary.
/// </summary>
public sealed class BindableDynamicDictionary : DynamicObject, INotifyPropertyChanged
{
/// <summary>
/// The internal dictionary.
/// </summary>
private readonly Dictionary<string, object> _dictionary;
/// <summary>
/// Creates a new BindableDynamicDictionary with an empty internal dictionary.
/// </summary>
public BindableDynamicDictionary()
{
_dictionary = new Dictionary<string, object>();
}
/// <summary>
/// Copies the contents of the given dictionary to initilize the internal dictionary.
/// </summary>
/// <param name="source"></param>
public BindableDynamicDictionary(IDictionary<string, object> source)
{
_dictionary = new Dictionary<string, object>(source);
}
/// <summary>
/// You can still use this as a dictionary.
/// </summary>
/// <param name="key"></param>
/// <returns></returns>
public object this[string key]
{
get
{
return _dictionary[key];
}
set
{
_dictionary[key] = value;
RaisePropertyChanged(key);
}
}
/// <summary>
/// This allows you to get properties dynamically.
/// </summary>
/// <param name="binder"></param>
/// <param name="result"></param>
/// <returns></returns>
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
return _dictionary.TryGetValue(binder.Name, out result);
}
/// <summary>
/// This allows you to set properties dynamically.
/// </summary>
/// <param name="binder"></param>
/// <param name="value"></param>
/// <returns></returns>
public override bool TrySetMember(SetMemberBinder binder, object value)
{
_dictionary[binder.Name] = value;
RaisePropertyChanged(binder.Name);
return true;
}
/// <summary>
/// This is used to list the current dynamic members.
/// </summary>
/// <returns></returns>
public override IEnumerable<string> GetDynamicMemberNames()
{
return _dictionary.Keys;
}
public event PropertyChangedEventHandler PropertyChanged;
private void RaisePropertyChanged(string propertyName)
{
var propChange = PropertyChanged;
if (propChange == null) return;
propChange(this, new PropertyChangedEventArgs(propertyName));
}
}
然后你可以像这样使用它:
private void testButton1_Click(object sender, RoutedEventArgs e)
{
// Creating a dynamic dictionary.
var dd = new BindableDynamicDictionary();
//access like any dictionary
dd["Age"] = 32;
//or as a dynamic
dynamic person = dd;
// Adding new dynamic properties.
// The TrySetMember method is called.
person.FirstName = "Alan";
person.LastName = "Evans";
//hacky for short example, should have a view model and use datacontext
var collection = new ObservableCollection<object>();
collection.Add(person);
dataGrid1.ItemsSource = collection;
}
Datagrid需要自定义代码来构建列:
XAML:
<DataGrid AutoGenerateColumns="True" Name="dataGrid1" AutoGeneratedColumns="dataGrid1_AutoGeneratedColumns" />
AutoGeneratedColumns事件:
private void dataGrid1_AutoGeneratedColumns(object sender, EventArgs e)
{
var dg = sender as DataGrid;
var first = dg.ItemsSource.Cast<object>().FirstOrDefault() as DynamicObject;
if (first == null) return;
var names = first.GetDynamicMemberNames();
foreach(var name in names)
{
dg.Columns.Add(new DataGridTextColumn { Header = name, Binding = new Binding(name) });
}
}
答案 1 :(得分:1)
基于westons回答我提出了另一个解决方案,而没有使用自定义的BindableDynamicDictionary类。
在命名空间ExpandoObject
中有一个名为System.Dynamic
的类(在ASP.NET中大量使用)。
它基本上与westons BindableDynamicDictionary做同样的事情,缺点是没有索引运算符,因为它显式实现了接口IDictionary<string, object>
private void MyDataGrid_AutoGeneratedColumns(object sender, EventArgs e)
{
var dg = sender as DataGrid;
dg.Columns.Clear();
var first = dg.ItemsSource.Cast<object>().FirstOrDefault() as IDictionary<string, object>;
if (first == null) return;
var names = first.Keys;
foreach (var name in names)
{
dg.Columns.Add(new DataGridTextColumn { Header = name, Binding = new Binding(name) });
}
}
请注意,唯一的区别是您必须通过索引运算符将ExpandoObject
强制转换为IDictionary<string, object>
来访问/添加值或属性。