在标题中,我有一个DataGrid和一个实现ICustomTypeDescriptor的ViewModel,在运行时添加了一些属性。
public class PartCloneSettingsController : BaseController, ICustomTypeDescriptor
{
...
private List<StringPropertyDescriptor> generatedProperties;
private void generateProperties()
{
foreach (PartAttributeDefinition item in PartAttributes.DefinedAttributes)
{
var propertyDescriptor = new StringPropertyDescriptor(item.AttributeTitle, typeof(PartCloneSettingsController), item);
// attach value changed handler [ memory leak? TODO: Remove handler at some point...]
propertyDescriptor.AddValueChanged(this, OnGeneratedPropertyChanged);
generatedProperties.Add(propertyDescriptor);
}
}
public PropertyDescriptorCollection GetProperties()
{
// Get All Default (defined) properties
var properties = TypeDescriptor.GetProperties(this, true);
// concat default properties and generated properties into a single collection
var newProperties = new PropertyDescriptorCollection(properties.Cast<PropertyDescriptor>().Concat(generatedProperties).ToArray());
return newProperties;
}
}
XAML中的DataGrid定义:
<DataGrid x:Name="dataGrid" AutoGenerateColumns="True" />
我像这样设置ItemsSource:
controller.LoadAssembly(ofd.FileName); // loads the file and creates a ObservableCollection of PartCloneSettingsControllers...
// set data grid source, doesn't create columns for generated properties...
overviewGrid.ItemsSource = controller.PartCloneSettingsControllers
// set data source from a winforms DataGridView which generates columns properly...
((System.Windows.Forms.DataGridView)wfhost.Child).DataSource = controller.PartCloneSettingsControllers;
其中controller.PartCloneSettingsControllers
定义为:
public ObservableCollection<PartCloneSettingsController> PartCloneSettingsControllers { get; private set; }
出于调试目的,我在Winforms控件主机中创建了一个DataGridView并将相同的ViewModel附加到它上面,并且:Winforms网格创建所有列并按我的意愿显示数据,但是WPF DataGrid无法生成列我的自定义属性(它适用于普通的普通属性)。
是否有人使用DataGrid,ICustomTypeDescriptor和AutoGenerateColumns = True有一个可行的解决方案(如果我在XAML中手动生成列,它可以正常工作,我可以绑定到我的所有属性......)
答案 0 :(得分:5)
如果PartCloneSettingsControllers
是除object
以外的某些项类型的泛型集合,它将使用泛型参数类型来填充列,而不是集合中项的运行时类型
例如,如果您的集合是IEnumerable<PartCloneSettingsController>
,那么网格将仅填充PartCloneSettingsController
类型(及其基本类型)上声明的属性的列< SUP> 1 。它不会检查集合中的实际对象,并且由于ICustomTypeDescriptor
在实例级别公开属性,因此网格将不会看到这些属性。
如果您可以选择在类型或集合级别而不是项目实例级别(例如,使用TypeDescriptionProvider
或ITypedList
)公开“动态”属性,则可能是您的最好的选择。否则,您将不得不使用网格无法推断项类型的集合类型(例如,List<object>
);这会强制网格检查它遇到的第一个项目,以确定列应该是什么。
1 网格最终使用TypeDescriptor.GetProperties(Type componentType)
解析属性,而不是TypeDescriptor.GetProperties(object component)
,因此如果没有要检查的实际项目,它无法知道哪个“动态”属性由各个项目公开。