Я пытаюсь использовать [TypeDescriptionProviderAttribute]
, чтобы дать моему классу дескриптор нестандартного типа. Это работает, но когда я реализую INotifyPropertyChanged
WPF, кажется, игнорирует дескриптор пользовательского типа и сразу переходит к свойству CLR (если оно существует). Вот фрагмент, позже я вставлю полный пример:
//[TypeDescriptionProvider(typeof(MyProvider))]
class MyModel : Object
//, INotifyPropertyChanged
//, ICustomTypeDescriptor
{
public string TheProperty { get { return "CLR - TheProperty"; } }
Я связываю TextBlock с TheProperty. Когда я ...
Оставьте все комментарии
Я вижу "CLR - TheProperty", как и ожидалось.
Использование [TypeDescriptionProvider]
Я вижу "MyPropertyDescriptor - TheProperty", как и ожидалось.
Использование ICustomTypeDescriptor
Я вижу "MyPropertyDescriptor - TheProperty", как и ожидалось.
Используйте ICustomTypeDescriptor
и INotifyPropertyChanged
Я вижу "MyPropertyDescriptor - TheProperty", как и ожидалось.
Используйте [TypeDescriptionProvider]
и INotifyPropertyChanged
Я вижу "CLR - TheProperty". Почему это так? Странно то, что пользовательские свойства без свойства CLR отображаются нормально. Мой дескриптор пользовательского типа также возвращает «MyPropertyDescriptor - AnotherProperty», который работает во всех случаях, поскольку не определено CLR AnotherProperty
.
В итоге, учитывая этот XAML
<StackPanel>
<TextBlock Text="{Binding TheProperty}" />
<TextBlock Text="{Binding AnotherProperty}" />
</StackPanel>
AnotherProperty
всегда работает должным образом, поскольку модель не имеет свойства CLR с именем "AnotherProperty". TheProperty
работает как положено , за исключением , когда используются [TypeDescriptionProvider]
и INotifyPropertyChanged
.
Вот полный код. Это немного долго, но большая часть этого не имеет значения, просто требуется System.ComponentModel
public partial class TestWindow : Window
{
public TestWindow()
{
InitializeComponent();
DataContext = new MyModel();
}
}
//[TypeDescriptionProvider(typeof(MyProvider))]
class MyModel : Object
//, INotifyPropertyChanged
//, ICustomTypeDescriptor
{
public string TheProperty { get { return "CLR - TheProperty"; } }
public event PropertyChangedEventHandler PropertyChanged;
public AttributeCollection GetAttributes()
{
return TypeDescriptor.GetAttributes(this);
}
public string GetClassName()
{
return TypeDescriptor.GetClassName(this);
}
public string GetComponentName()
{
return TypeDescriptor.GetComponentName(this);
}
public TypeConverter GetConverter()
{
return TypeDescriptor.GetConverter(this);
}
public EventDescriptor GetDefaultEvent()
{
return TypeDescriptor.GetDefaultEvent(this);
}
public PropertyDescriptor GetDefaultProperty()
{
return TypeDescriptor.GetDefaultProperty(this);
}
public object GetEditor(Type editorBaseType)
{
return TypeDescriptor.GetEditor(this, editorBaseType);
}
public EventDescriptorCollection GetEvents(Attribute[] attributes)
{
return TypeDescriptor.GetEvents(this, attributes);
}
public EventDescriptorCollection GetEvents()
{
return TypeDescriptor.GetEvents(this);
}
public PropertyDescriptorCollection GetProperties(Attribute[] attributes)
{
return TypeDescriptor.GetProperties(this, attributes);
}
public PropertyDescriptorCollection GetProperties()
{
return MyTypeDescriptor.GetCustomProperties();
}
public object GetPropertyOwner(PropertyDescriptor pd)
{
return this;
}
}
class MyProvider : TypeDescriptionProvider
{
public override ICustomTypeDescriptor GetTypeDescriptor(Type objectType, object instance)
{
return new MyTypeDescriptor();
}
}
class MyTypeDescriptor : CustomTypeDescriptor
{
public override PropertyDescriptorCollection GetProperties()
{
return GetCustomProperties();
}
public static PropertyDescriptorCollection GetCustomProperties()
{
return new PropertyDescriptorCollection(
new[] {
new MyPropertyDescriptor("TheProperty"),
new MyPropertyDescriptor("AnotherProperty")
});
}
}
class MyPropertyDescriptor : PropertyDescriptor
{
public MyPropertyDescriptor(string propName)
: base(propName, null)
{
}
public override bool CanResetValue(object component)
{
return false;
}
public override Type ComponentType
{
get { return typeof(MyModel); }
}
public override object GetValue(object component)
{
return "MyPropertyDescriptor - " + Name;
}
public override bool IsReadOnly
{
get { return true; }
}
public override Type PropertyType
{
get { return typeof(string); }
}
public override void ResetValue(object component)
{
throw new InvalidOperationException("cannot reset value");
}
public override void SetValue(object component, object value)
{
throw new InvalidOperationException("property is readonly");
}
public override bool ShouldSerializeValue(object component)
{
return true;
}
}