January 29th, 2021

Beautiful Custom RadioButton with Xamarin.Forms 5

James Montemagno
Principal Manager, Tech PM

This weekend I was looking to update my settings page on my latest app, My Cadence, and introduce a new option to allow the user to select a default theme for their application. There are a few ways to implement this features. In previous applications I have built, I have used the Picker control to give users the option to pick a setting:

Picker control showing a list of theme options

This is a pretty good experience, however I wanted to see if there was something new in Xamarin.Forms 5 that I could leverage to create a better experience. Browsing the documentation I stumbled upon the brand new RadioButton control that is fully templatable! This means you can completely customize the look and feel of the control while still maintaining the functionality of the control.

So, let’s take a look at how to leverage this new feature to create a new way for my users to select the theme.

Enter RadioButton!

The first thing to do is to add the RadioButton to your application. To create a theme selection UI we will create a Grid with 3 columns each with a RadioButton in it.

<Grid
    ColumnDefinitions="*,*,*"
    ColumnSpacing="10">
    <RadioButton IsChecked="True" Content="System Default"/>
    <RadioButton Grid.Column="1" Content="Light"/>
    <RadioButton Grid.Column="2" Content="Dark"/>
</Grid>

This will result with the default control that displays the RadioButton besides text.

Default radio buttons with text

One of the beautiful parts of the new RadioButton is that it has control templating built in. This means you can override the default template that is a Label with anything!

Control Template

In our ContentPage.Resources we can add a new control template that can be applied to the RadioButton. It controls what the RadioButton looks like and even has access to the VisualStateManager so we can customize what the Checked state looks like. Now, this is a bit of code, but what we are creating in a template where we are drawing the circle in the top right of a frame that has a ContentPresenter in the main area in which we can fill in any UI we want.

   <ContentPage.Resources>
        <ResourceDictionary>

            <Color x:Key="LightRadioButtonColor">#F3F2F1</Color>
            <Color x:Key="DarkRadioButtonColor">#9B9A99</Color>

            <ControlTemplate x:Key="ThemeRadioTemplate">
                <Frame
                    Padding="0"
                    BackgroundColor="{AppThemeBinding Dark={StaticResource DarkRadioButtonColor},
                                                      Light={StaticResource LightRadioButtonColor}}"
                    BorderColor="{AppThemeBinding Dark={StaticResource DarkRadioButtonColor},
                                                  Light={StaticResource LightRadioButtonColor}}"
                    HasShadow="False"
                    HeightRequest="80"
                    HorizontalOptions="Start"
                    VerticalOptions="Start"
                    WidthRequest="80">
                    <Grid Margin="4" WidthRequest="80">
                        <Grid
                            HeightRequest="20"
                            HorizontalOptions="End"
                            VerticalOptions="Start"
                            WidthRequest="20">
                            <Ellipse
                                Fill="White"
                                HeightRequest="18"
                                HorizontalOptions="Center"
                                Stroke="#2E2545"
                                StrokeThickness="1"
                                VerticalOptions="Center"
                                WidthRequest="18" />
                            <Ellipse
                                x:Name="Check"
                                BackgroundColor="Transparent"
                                Fill="#df6e57"
                                HeightRequest="10"
                                HorizontalOptions="Center"
                                Stroke="#df6e57"
                                StrokeThickness="0"
                                VerticalOptions="Center"
                                WidthRequest="10" />
                        </Grid>
                        <!-- This enables us to put in dynamic content -->
                        <ContentPresenter />
                    </Grid>
                    <VisualStateManager.VisualStateGroups>
                        <VisualStateGroupList>
                            <VisualStateGroup x:Name="CheckedStates">
                                <VisualState x:Name="Checked">
                                    <VisualState.Setters>
                                        <Setter Property="BorderColor" Value="#df6e57" />
                                        <Setter TargetName="Check" Property="Opacity" Value="1" />
                                    </VisualState.Setters>
                                </VisualState>
                                <VisualState x:Name="Unchecked">
                                    <VisualState.Setters>
                                        <Setter Property="BorderColor" Value="#F3F2F1" />
                                        <Setter TargetName="Check" Property="Opacity" Value="0" />
                                    </VisualState.Setters>
                                </VisualState>
                            </VisualStateGroup>
                        </VisualStateGroupList>
                    </VisualStateManager.VisualStateGroups>
                </Frame>
            </ControlTemplate>
        </ResourceDictionary>
    </ContentPage.Resources>

We can now apply this template to each RadioButton. It will continue to use our text label content but apply the control template that we just setup.

<Grid ColumnDefinitions="*,*,*" ColumnSpacing="10">
    <RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" Content="Default" IsChecked="True" />
    <RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" Grid.Column="1" Content="Light" />
    <RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" Grid.Column="2" Content="Dark" />
</Grid>

Radio buttons using template with just labels

We aren’t done though as we can now leverage the ContentPresenter and set the Content to anything that we want. For this UI I want to put in an Image (in this case a FontAwesome label) and text on the bottom.

<RadioButton.Content>
    <StackLayout HorizontalOptions="Center" VerticalOptions="Center">
        <Image Scale="0.75" Source="{FontImage FontFamily=FA, Glyph={x:Static helpers:FAIcons.UserCog}, Color=#323130}" />
        <Label FontSize="Small" Text="System" TextColor="#323130" />
    </StackLayout>
</RadioButton.Content>

The full XAML for all three RadioButton controls would look a bit like this:

<RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" IsChecked="True">
    <RadioButton.Content>
        <StackLayout HorizontalOptions="Center" VerticalOptions="Center">
            <Image Scale="0.75" Source="{FontImage FontFamily=FA, Glyph={x:Static helpers:FAIcons.UserCog}, Color=#323130}" />
            <Label FontSize="Small" Text="System" TextColor="#323130" />
        </StackLayout>
    </RadioButton.Content>
</RadioButton>
<RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" Grid.Column="1">
    <RadioButton.Content>
        <StackLayout HorizontalOptions="Center" VerticalOptions="Center">
            <Image Scale="0.75" Source="{FontImage FontFamily=FA, Glyph={x:Static helpers:FAIcons.Sun}, Color=#323130}" />
            <Label FontSize="Small" Text="Light" TextColor="#323130" />
        </StackLayout>
    </RadioButton.Content>
</RadioButton>
<RadioButton ControlTemplate="{StaticResource ThemeRadioTemplate}" Grid.Column="2">
    <RadioButton.Content>
        <StackLayout HorizontalOptions="Center" VerticalOptions="Center">
            <Image Scale="0.75" Source="{FontImage FontFamily=FA, Glyph={x:Static helpers:FAIcons.Moon}, Color=#323130}" />
            <Label FontSize="Small" Text="Dark" TextColor="#323130" />
        </StackLayout>
    </RadioButton.Content>
</RadioButton>

This will result in great cross-platform custom RadioButton!

Final radio buttons with icons and text

Learn More

Checkout the full documentation for RadioButton and also control templating for more insight on how to implement this in your app.

You can also browse the source code of LosGatos on GitHub, which is a delightful sample from David Ortinau that uses this technique.

Author

James Montemagno
Principal Manager, Tech PM

James Montemagno is a Principal Lead Program Manager for Developer Community at Microsoft. He has been a .NET developer since 2005, working in a wide range of industries including game development, printer software, and web services. Prior to becoming a Principal Program Manager, James was a professional mobile developer and has now been crafting apps since 2011 with Xamarin. In his spare time, he is most likely cycling around Seattle or guzzling gallons of coffee at a local coffee shop. He ...

More about author

3 comments

Discussion is closed. Login to edit/delete existing comments.

  • John

    Will XF 5 support grpc?Really look forward to this happen

  • Ernest L.

    Good work James. Ability to provide custom ControlTemplate like in WPF is definitely useful.
    I noticed the VisualState.Setters with a target control does not accept the actual control target property.
    It would have been easier to set Fill/Stroke of the Ellipse directly.

  • Robert Burton

    Hi James! Thank you very much for this…excellent timing for me. However, in the Control Template Visual State on the BorderColor Property, Visual studio is giving me an error that it is Not Recognized or Not Accessible. I’m using Visual Studio preview 16.9 Preview 3 and have updated all the Nuget packages are up to date.

    Do you know what might be causing this? It stops Hot Reload from working also when run.