Avalonia UI – How to bind to direct properties of a user control within the user control itself
Published:
Modified:
- 1 Introduction
- 2 Source Code
- 3 Typical beginner approach (does not work)
- 4 User Control – Code Behind
- 5 User Control – XAML – Defining user control
- 6 User Control – XAML – Bind to user control by name
- 7 User Control – XAML – Bind to user control using ancestor type syntax
- 8 User Control – XAML – Bind to user control by setting data context
- 9 User control binding in action
- 10 Summary
Introduction
Avalonia UI has powerful data binding mechanism. Avalonia UI allows us to reuse our code by creating custom user controls. So what’s the problem?! The problem is that, for beginners, using data binding within their user controls is not straightforward. This article explains how this is done.
Source Code
The complete source code is available in my GitHub repository.
The source code shows how to:
- Create a simple user control
- Define a
DirectProperty
in the user control code behind file - Access the
DirectProperty
from the user control XAML file - Use the user control from the main view
The source code will teach you how to:
- Bind to a named control (WPF/UWP style and Avalonia UI style)
- Bind to an ancestor
- Set data context and use simple binding
Typical beginner approach (does not work)
When creating a user control, this is the typically scenario:
- We have our main window, main view and our custom user control.
- Main window contains the main view, and the main view contains the user control.
- Main window and main view are using a view model as their data context.
- The user control automatically inherits data context from its parent (the view model).
- Naturally, within the user control, we try to bind to a direct property of the user control itself. But it doesn’t work!
- To make it work, we can either set a data context of the user control and use simple bindings (I’ll write about this in a separate article), or we can use a bit more complex bindings.
User Control – Code Behind
In our simple code behind, we simply have one simple direct property:
public static readonly DirectProperty<BindableUserControl, string> GreetingProperty =
AvaloniaProperty.RegisterDirect<BindableUserControl, string>(
nameof(Greeting),
o => o.Greeting,
(o, v) => o.Greeting = v
);
private string greeting;
public string Greeting
{
get => greeting;
set => SetAndRaise(GreetingProperty, ref greeting, value);
}
User Control – XAML – Defining user control
There are 2 important things to mention here:
- We use the
x:DataType
XAML attribute because we want to enable compiled bindings. Compiled bindings save your time by making it easier for the editor to make useful suggestions to you, and by catching syntax errors at the compilation/build time. Compiled bindings also improve the performance of the application by eliminating the use of reflection. - We’ll use data binding by name, so we need to define the user control name (via the
Name
XAML attribute).
<UserControl xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:ctrl="clr-namespace:Monsalma_AvaloniaUserControlBinding.Controls"
x:Class="Monsalma_AvaloniaUserControlBinding.Controls.BindableUserControl"
x:DataType="ctrl:BindableUserControl"
Name="BindableUserControl1">
User Control – XAML – Bind to user control by name
To bind to our custom user control from the control itself, we can use the ElementName
(WPF/UWP style) or the name prefix character (#
).
WPF/UWP style – ElementName
<TextBlock
Text="{Binding Greeting, ElementName=BindableUserControl1}" />
Avalonia UI style – Name prefix (#)
<TextBlock
Text="{Binding #BindableUserControl1.Greeting}" />
User Control – XAML – Bind to user control using ancestor type syntax
There are a few ways to bind to an ancestor:
$parent
keyword can be used to reference the parent control (in logical, not visual tree).$parent[1]
will go 2 levels up,$parent[2]
3 levels up and so on. This implies that$parent[0]
is equivalent to$parent
.- In our demo, we reference the ancestor using the ancestor type syntax –
$parent[ctrl:BindableUserControl]
. - Avalonia UI also allows to combine the level index and ancestor type. In some case we could use
$parent[ctrl:BindableUserControl;1]
.
<TextBlock
Text="{Binding $parent[ctrl:BindableUserControl].Greeting}" />
User Control – XAML – Bind to user control by setting data context
Our last option is to define DataContext
for the TextBlock
control. Doing so will override the hierarchical data context, which is MainViewModel
in our case, and will allow us to use the simplest form of data binding ({Binding Greeting}
). Note that we are referencing the user control by name (Avalonia UI syntax – #
prefix).
<TextBlock
DataContext="{Binding #BindableUserControl1}"
Text="{Binding Greeting}" />
User control binding in action
For this demo, I chose Windows Desktop and Android as targets. Let’s see the user control data binding in action.
Windows Desktop
data:image/s3,"s3://crabby-images/36de3/36de31ba2a2df13921938b2f769e59b404f5f5c6" alt="Avalonia UI - User Control Data Binding Demo - Windows"
Android Emulator – Google Pixel 3 XL (API 30)
data:image/s3,"s3://crabby-images/df214/df2142458ba35635442490adaec468d746da26b8" alt="Avalonia UI - User Control Data Binding Demo - Android"
Summary
Avalonia UI is great. Its data binding mechanism is great. Its way of defining user controls is great. I hope this article helps to explain the way data bindings and user controls can be used together.
Leave a Reply