WPF makes the process of data binding UI controls to the properties of objects in the code-behind simple and intuitive. Unfortnately, C# doesn't make it quite as easy to bind objects in the code behind to other objects. (i.e. binding the properties of your model objects to the corresponding properties of your view model object)
In order to bind the properties of two objects, the following is required:
- Both the source and target objects must implement INotifyPropertyChanged
- The properties to be data bound must be publicly accessible properties of their respective objects
To bind together the properties of a source and target object, simply invoke the static CreateBinding method of the BindingManager class:
int bindingId = BindingManager.Register(sourceObject, s => s.SourceProperty, targetObject, t => t.TargetProperty, fallback, bindingMode, converter);
The fallback
parameter specifies the value to be used in updating the target if the source property has no data or when conversion to the target type provides no result. In a two-way binding, this value is also used to update the source when the target has no value or when converting the target value provides no result. This parameter is optional and if omitted, will cause the default of the source property's type to be used as a fallback. For many types, the default is null, so failing to specify a fallback may lead to null reference exceptions.
The bindingMode
parameter is a member of the BindingMode
enum. You can specify the following:
BindingMode.TwoWay
(The default if no argument is provided)BindingMode.OneWay
BindingMode.OneWayToSource
The converter
parameter is an object which implements the IBindingConverter
interface, which is part of the framework. You may opt not to provide a converter, but if no converter is specified and the source and target properties are of incompatible types, an exception will be raised.
The Register()
method returns an integer key which is used to track and manage the binding internally. You can use this key later to dispose of the binding by providing it to the static Unregister()
method.
To remove a data binding, invoke the static Unregister() method and provide the integer key of the binding to be destroyed:
BindingManager.Unregister(bindingId);
Note that binding to nested properties does work (ie binding outer.inner.property to someObject.property), provided that:
- inner implements INotifyPropertyChanged (it is not actually necessary for outer to implement INotifyPropertyChanged in this instance)
- the call to CreateBinding is
BindingManager.CreateBinding(outer.inner, i => i.property, someObject, someObject.property...)
In other words, attempting to bind as such:
BindingManager.CreateBinding(outer, o => o.inner.property, target, t => t.Property...)
will not work, and although I am working on making this work, it is not high on my list of priorities.
To understand why it makes sense that this binding does not work, consider:
- In order for outer to notify the binding manager that a property of inner has changed, inner would need to notify outer that said property has changed
- In order for inner to notify outer, inner would need to implement INotifyPropertyChanged and outer would need to subscribe to inners NotifyPropertyChanged event
Thus, in any scenario involving a nested binding, the inner object whose property is to be bound must implement INotifyPropertychanged, which means we may as well bind directly to inner rather than going through its containing object first.
When a call to BindingManager.Register()
occurs, the BindingManager attempts to create an instance of TypedBinding. TypedBinding receives INotifyPropertyChanged notifications from the source and target and updates the corresponding properties as required.
The TypedBinding class manages the work of responding to INotifyPropertyChanged events and updating the source and target as required. When an INotifyPropertyChanged event is handled, delegates to the appropriate getter/setter methods are invoked to retrieve the changed value and update the corresponding property on the opposing object.
Whenever a INotifyPropertyChanged event is raised, the TypedBinding checks if the source and target objects are still alive. If either has been garbage collected, the TypedBinding class sets itself inactive and requests disposal from the BindingManager.
Neither the TypedBinding nor the BindingManager classes store strong references to the source or target objects. References to the source and target object are maintained as WeakReference objects. Delegates for the get/set methods of properties are cached as instance-free delegates. Finally, the INotifyPropertyChanged notifications are handled via the PropertyChangedEventManager. Thus, the existence of a binding between two objects should not prevent either from being garbage collected.
Currently SimpleBinding does not support multiple bindings, such as are possible with the WPF Property system. Only binding once source property to one target property is supported.
Nested Property Binding. Currently, it is possible to bind to a nested property (i.e. outerObject.InnerObject.Property), but only by conforming to the requirements specified above. It may or may not be possible to allow for more flexible binding procedures later.