How notification works for Attached Properties

208 Views Asked by At

I am trying to understand how notification works for Attached Properties in WPF.

For example, consider the ScrollViewer.CanContentScrollProperty property.

Suppose we have the following ListBox

<ListBox x:Name="MainListBox"
         Grid.Row="1"
         ScrollViewer.CanContentScroll="False">
  <ListBox.ItemTemplate>
    <DataTemplate>
      <StackPanel Orientation="Horizontal">
        <TextBlock Text="{Binding CustomerID}"/>
      </StackPanel>
    </DataTemplate>
  </ListBox.ItemTemplate>
</ListBox>

And in the Window code we will write the following code

private void Button_Click_1(object sender, RoutedEventArgs e)
{
  MainListBox.SetValue(ScrollViewer.CanContentScrollProperty, true);
}

After executing Button_Click_1, the ListBox will automatically switch to CanContentScroll = true mode, but it is not clear how ScrollViewer receives information that the values of the connected property have been changed. It can be assumed that through the function FrameworkPropertyMetadata (PropertyChangedCallback propertyChangedCallback) , but the fact is that the property ScrollViewer.CanContentScrollProperty does not define a PropertyChanged function.

Here is the declaration code for this property.

      public class ScrollViewer : ContentControl
...
        public static readonly DependencyProperty CanContentScrollProperty = DependencyProperty.RegisterAttached(
nameof (CanContentScroll), typeof (bool), typeof (ScrollViewer), 
(PropertyMetadata) new FrameworkPropertyMetadata(BooleanBoxes.FalseBox));

It can be seen that the function for the change is not registered.

How then does ScrollViewer know that a ScrollViewer.CanContentScrollProperty property has been changed?

2

There are 2 best solutions below

0
On

Unfortunately, it was a while the last time I played with WPF but from what I remember/found they don't use INotifyPropertyChanged.

INotifyPropertyChanged is intended to broadcast changes in a model only (on your data).

UI controls inherit from DependencyObject class - which is one part of how their magic comes from. The other part is that UI control's properties use DependencyProperty (like your attached properties).

Some quotes from https://learn.microsoft.com/en-us/dotnet/desktop-wpf/data/data-binding-overview :

The target property must be a dependency property. Most UIElement properties are dependency properties, and most dependency properties, except read-only ones, support data binding by default. (Only types derived from DependencyObject can define dependency properties; and all UIElement types derive from DependencyObject.)

and

To detect source changes (applicable to OneWay and TwoWay bindings), the source must implement a suitable property change notification mechanism such as INotifyPropertyChanged. See How to: Implement property change notification for an example of an INotifyPropertyChanged implementation.

Then you can found :

Although not shown in the figure, it should be noted that the binding source object is not restricted to being a custom .NET object. WPF data binding supports data in the form of .NET objects and XML. To provide some examples, your binding source may be a UIElement, any list object, an ADO.NET or Web Services object, or an XmlNode that contains your XML data. For more information, see Binding sources overview.

In the Binding sources overview, you can read that INotifyCollectionChanged and INotifyPropertyChange are the interfaces to implement to support notification.

For ADO.NET objects IBindingList is the interface doing the job. (this would be the other documented binding method)

For CLR you still need to implement INotifyCollectionChanged (they refer you to the "Implementing a Class for the Binding Source chapter")

As for DependencyObject well... they do it by themself.

1
On

On the call stack, we can assume that a change in the value of a property is caught through some kind of XAML trigger and assigned to the ScrollContentPresenter.CanContentScrollProperty property

enter image description here