Skip to content

Latest commit

 

History

History
219 lines (169 loc) · 8.59 KB

DependencyProperty.md

File metadata and controls

219 lines (169 loc) · 8.59 KB

⚙️ DependencyProperties with SourceGeneration

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.


🚀 Setting Up Your First UserControl

✨ Creating a Simple UserControl

Let's start by defining a UserControl using source generators.

[DependencyProperty<bool>("IsRunning")]
public partial class TestView
{
    public TestView()
    {
        InitializeComponent();
    }
}

🔍 What's Happening Here?

  • The DependencyPropertyAttribute automatically generates the IsRunning property, including INotifyPropertyChanged support.

🖥️ XAML Binding Example

<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.


📌 Summary

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

🚀 Why Use Atc.Wpf Source Generators?

  • Eliminates boilerplate – Just declare the property, and the generator handles the rest.

🔎 Behind the scenes

📝 Human-Written Code - for simple example

[DependencyProperty<bool>("IsRunning"]
public partial class MyControl : UserControl
{
}

In this example:

  • The [DependencyProperty<bool>("IsRunning")] attribute indicates that a dependency property named IsRunning of type bool should be generated for the MyControl class.

⚙️ Auto-Generated Code - for simple example

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);
    }
}

📝 Human-Written Code - for complex example

[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 type bool for MyControl.
    • The source generator will automatically create a DependencyProperty field and a CLR property wrapper.
  • DefaultValue = false

    • Specifies the default value of IsRunning as false.
    • This means that when an instance of MyControl is created, IsRunning will be false unless explicitly set.
  • 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 be true under specific circumstances, the CoerceValueCallback 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 of BindingExpression.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 modify IsRunning.
    • 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 a bool:
      • true: The value is accepted and applied to the property.
      • false: The value is considered invalid, and an exception is thrown.

⚙️ Auto-Generated Code - for complex example

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);
    }
}