I have a control which inherits from (you guessed it) Control.
I want to receive a notification whenever the FontSize
or Style
properties are changed. In WPF, I would do that by calling DependencyProperty.OverrideMetadata()
. Of course, useful things like that have no place in Silverlight. So, how might one receive those kinds of notifications?

- 12,256
- 14
- 53
- 61
5 Answers
I think here is a better way. Still need to see the pros and Cons.
/// Listen for change of the dependency property
public void RegisterForNotification(string propertyName, FrameworkElement element, PropertyChangedCallback callback)
{
//Bind to a depedency property
Binding b = new Binding(propertyName) { Source = element };
var prop = System.Windows.DependencyProperty.RegisterAttached(
"ListenAttached"+propertyName,
typeof(object),
typeof(UserControl),
new System.Windows.PropertyMetadata(callback));
element.SetBinding(prop, b);
}
And now, you can call RegisterForNotification to register for a change notification of a property of an element, like .
RegisterForNotification("Text", this.txtMain,(d,e)=>MessageBox.Show("Text changed"));
RegisterForNotification("Value", this.sliderMain, (d, e) => MessageBox.Show("Value changed"));
See my post here on the same http://amazedsaint.blogspot.com/2009/12/silverlight-listening-to-dependency.html
Using Silverlight 4.0 beta.

- 31,361
- 18
- 86
- 116

- 7,642
- 7
- 54
- 83
-
I like that this is a nicer way of wrapping up that ugly binding, but that is an awfully heavy way to get property change notifications. – MojoFilter Dec 03 '09 at 13:40
-
1As long as we need to do it entirely from the code, I guess we don't have another option till we get the DependencyPropertyDescriptor in Silverlight from MS – amazedsaint Dec 04 '09 at 11:13
-
I've also used this approach, unfortunately it leaks memory. In my experience it will hold a reference to what ever element you use it with, which will in turn hold on to other objects. The culprit appears to be the callback which will retain a reference to your view for the life of your application. – Ian Oakes Dec 13 '10 at 04:07
-
It works (+1!) but as Ian said its not optimal. As a workaround I've moved the attached property (in my case called `ListenAttachedIsEnabledProperty` to the class I want to override and added the following two lines in the constructor `var b = new Binding("IsEnabled") { Source = this }; SetBinding(ListenAttachedIsEnabledProperty, b);` – Dr. Andrew Burnett-Thompson Mar 01 '12 at 16:10
-
it will throw when calling RegisterForNotification for multiple textboxes probably because you can't create multiple DependencyProperties with the same Name. – Firo May 27 '13 at 13:02
It's a rather disgusting hack, but you could use a two-way binding to simulate this.
i.e. have something like:
public class FontSizeListener {
public double FontSize {
get { return fontSize; }
set { fontSize = value; OnFontSizeChanged (this, EventArgs.Empty); }
}
public event EventHandler FontSizeChanged;
void OnFontSizeChanged (object sender, EventArgs e) {
if (FontSizeChanged != null) FontSizeChanged (sender, e);
}
}
then create the binding like:
<Canvas>
<Canvas.Resources>
<FontSizeListener x:Key="listener" />
</Canvas.Resources>
<MyControlSubclass FontSize="{Binding Mode=TwoWay, Source={StaticResource listener}, Path=FontSize}" />
</Canvas>
then hook up to the listener's event in your control subclass.

- 416
- 4
- 8
-
When the MyControlSubclass FontSize property change, the binding will be clear so it seems it doesn't work? not? – Amir Karimi Aug 22 '11 at 22:55
You cannot externally listen to dependency property changed notifications.
You can access the Dependency Property Metadata with the following line of code:
PropertyMetadata metaData = Control.ActualHeightProperty.GetMetadata(typeof(Control));
However, the only public member that is exposed is "DefaultValue".
There are a multitude of ways to do this in WPF. But they are currently not supported by Silverlight 2 or 3.

- 2,783
- 1
- 23
- 30
The only solution I see is to listen to the LayoutUpdated event - yes, I know it is called a lot. Note however that in some cases it won't be called even though FontSize or Style has changed.

- 21
- 1
This is what I always use (haven't tested it on SL though, just on WPF):
/// <summary>
/// This method registers a callback on a dependency object to be called
/// when the value of the DP changes.
/// </summary>
/// <param name="owner">The owning object.</param>
/// <param name="property">The DependencyProperty to watch.</param>
/// <param name="handler">The action to call out when the DP changes.</param>
public static void RegisterDepPropCallback(object owner, DependencyProperty property, EventHandler handler)
{
// FIXME: We could implement this as an extension, but let's not get
// too Ruby-like
var dpd = DependencyPropertyDescriptor.FromProperty(property, owner.GetType());
dpd.AddValueChanged(owner, handler);
}

- 73,868
- 16
- 141
- 209
-
5I'm pretty sure DependencyPropertyDescrtiptors are mythical beasts in Silverlight, unfortunately. – MojoFilter May 07 '09 at 15:12
-
2