In WPF, dependency properties are a specialized type of property that extends the functionality of standard CLR properties. They support features such as data binding, animation, and property value inheritance, which are integral to the WPF property system. However, defining dependency properties traditionally involves verbose boilerplate code. To streamline this process, source generators can automatically generate the necessary code, reducing errors and improving maintainability.
Let's start by defining a UserControl using source generators.
[DependencyProperty<bool>("IsRunning")]
public partial class TestView
{
public TestView()
{
InitializeComponent();
}
}
- The
DependencyPropertyAttribute
automatically generates theIsRunning
property, includingINotifyPropertyChanged
support.
<UserControl xmlns:local="clr-namespace:MyApp.MyUserControl"
x:Name="UcMyUserControl">
<StackPanel>
<atc:LabelToggleSwitch
IsOn="{Binding ElementName=UcMyUserControl, Path=IsRunning}"
LabelText="Start / Stop spinner" />
<atc:BusyOverlay IsBusy="{Binding ElementName=UcMyUserControl, Path=IsRunning}">
<TextBox Text="Hello world" />
</atc:BusyOverlay>
</StackPanel>
</UserControl>
This setup allows the UI to dynamically update when the IsRunning property changes.
This example showcases advanced metadata for dependency properties, allowing:
- ✔️ Automatic property change notifications
- ✔️ Value coercion and validation
- ✔️ Optimized UI performance with layout invalidation
- ✔️ Flexible data binding behavior
- ✔️ Control over animation support
- ✅ Eliminates boilerplate – Just declare the property, and the generator handles the rest.
[DependencyProperty<bool>("IsRunning"]
public partial class MyControl : UserControl
{
}
In this example:
- The
[DependencyProperty<bool>("IsRunning")]
attribute indicates that a dependency property namedIsRunning
of typebool
should be generated for theMyControl
class.
The source generator will produce code equivalent to:
public partial class MyControl
{
public static readonly DependencyProperty IsRunningProperty = DependencyProperty.Register(
nameof(IsRunning),
typeof(bool),
typeof(MyControl),
new FrameworkPropertyMetadata(defaultValue: BooleanBoxes.TrueBox);
public bool IsRunning
{
get => (bool)GetValue(IsRunningProperty);
set => SetValue(IsRunningProperty, value);
}
}
[DependencyProperty<bool>(
"IsRunning",
DefaultValue = false,
PropertyChangedCallback = nameof(PropertyChangedCallback),
CoerceValueCallback = nameof(CoerceValueCallback),
Flags = FrameworkPropertyMetadataOptions.AffectsMeasure | FrameworkPropertyMetadataOptions.AffectsRender,
DefaultUpdateSourceTrigger = UpdateSourceTrigger.Default,
IsAnimationProhibited = true,
ValidateValueCallback = nameof(ValidateValueCallback))]
public partial class MyControl : UserControl
{
private static void PropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
throw new NotImplementedException();
}
private static object CoerceValueCallback(DependencyObject d, object baseValue)
{
throw new NotImplementedException();
}
private static bool ValidateValueCallback(object value)
{
throw new NotImplementedException();
}
}
In this example:
-
[DependencyProperty<bool>("IsRunning")]
- Declares a dependency property named
IsRunning
of typebool
forMyControl
. - The source generator will automatically create a
DependencyProperty
field and a CLR property wrapper.
- Declares a dependency property named
-
DefaultValue = false
- Specifies the default value of
IsRunning
asfalse
. - This means that when an instance of
MyControl
is created,IsRunning
will befalse
unless explicitly set.
- Specifies the default value of
-
PropertyChangedCallback = nameof(PropertyChangedCallback)
- Assigns a property changed callback method, which is invoked whenever the property's value changes.
- In this example,
PropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs e)
is defined as a static method. - This method allows you to respond to property changes, such as triggering UI updates or executing business logic.
-
CoerceValueCallback = nameof(CoerceValueCallback)
- Assigns a coerce value callback method, which is called before setting the property’s value.
- This function allows validation, restricting the range of acceptable values, or adjusting the value based on other conditions.
- For instance, if
IsRunning
should never betrue
under specific circumstances, theCoerceValueCallback
could enforce that rule.
-
Flags = FrameworkPropertyMetadataOptions.AffectsMeasure | FrameworkPropertyMetadataOptions.AffectsRender
- Configures how the property affects the WPF layout system when it changes.
AffectsMeasure
: Triggers a re-measure of the UI element if this property changes.AffectsRender
: Causes a re-render of the control when this property is modified.- These options ensure that any UI elements depending on
IsRunning
will recalculate their size and appearance accordingly.
-
FrameworkPropertyMetadataOptions
- A flag-based enumeration used to specify additional property behavior in WPF.
- Other possible values include:
AffectsParentMeasure
: Causes a layout pass on the parent when the property changes.AffectsArrange
: Forces an arrange pass when the property changes.Inherits
: Allows the property value to propagate down the visual tree.BindsTwoWayByDefault
: Sets the default binding mode to TwoWay.
- These flags optimize performance by ensuring layout changes only occur when necessary.
-
DefaultUpdateSourceTrigger = UpdateSourceTrigger.Default
- Specifies how data binding updates the property's source.
- The
Default
value means that the default behavior of the property type is used. - Other possible values:
PropertyChanged
: Updates the source immediately when the property changes.LostFocus
: Updates the source when the control loses focus (e.g., leaving a text box).Explicit
: Requires manual invocation ofBindingExpression.UpdateSource()
.
-
IsAnimationProhibited = true
- Prevents animations from affecting this property.
- Some dependency properties allow animations to change their values smoothly over time.
- By setting
IsAnimationProhibited = true
, you ensure that no animations can modifyIsRunning
. - This is useful for properties where instant updates are required, such as boolean state changes.
-
ValidateValueCallback = nameof(ValidateValueCallback)
- Assigns a validation callback method, which ensures that only valid values are assigned to the dependency property.
- This function executes before the property value is set, allowing you to reject invalid values before they are applied.
- The
ValidateValueCallback
method should return abool
:true
: The value is accepted and applied to the property.false
: The value is considered invalid, and an exception is thrown.
The source generator will produce code equivalent to:
public partial class MyControl
{
public static readonly DependencyProperty IsRunningProperty = DependencyProperty.Register(
nameof(IsRunning),
typeof(bool),
typeof(MyControl),
new FrameworkPropertyMetadata(
defaultValue: BooleanBoxes.FalseBox,
propertyChangedCallback: PropertyChangedCallback,
coerceValueCallback: CoerceValueCallback,
flags: FrameworkPropertyMetadataOptions.AffectsMeasure | FrameworkPropertyMetadataOptions.AffectsRender,
defaultUpdateSourceTrigger: UpdateSourceTrigger.Default,
isAnimationProhibited: true,
validateValueCallback = ValidateValueCallback));
public bool IsRunning
{
get => (bool)GetValue(IsRunningProperty);
set => SetValue(IsRunningProperty, value);
}
}