Many users of JQuery UI (both Mobile as well as web) are quite fond of accordion user control and ask for the same or a similar kind of control in Xamarin/ Xamarin Forms. As we all know that Xamarin forms controls are an abstraction to the native controls available in respective native platforms and since there is no accordion control available in any of the native mobile frameworks it’s not available as an out-of-the-box control in Xamarin Forms. So in this article we will be creating a simple accordion user-control using simple Xamarin Forms controls like Button and ContentView.
Firstly, we need to understand the functionality of an accordion, as per Wikipedia the developer's definition of an accordion is:
Several buttons or labels are stacked upon one another. At most one of them can be “active”. When a button is active the space below the button is used to display a paned window. The pane is usually constrained by the width of labels. When opened it shifts labels under the clicked label down according to the height of that window. Only one button or pane combination can be active at any one time; when a button is selected any other active panes cease to be active and are hidden. The active pane may have scrollbars.
And that’s exactly what I have done.
The ‘
Accordion’ class is extended from
ContentView class of Xamarin; forms in which I am creating an accordion from list ‘
AccordianSource’ objects on DataBind Method of the Class. The code of ‘Accordion’ user control is as follows:
This control has one public method ‘
DataBind’ and two public properties ‘
FirstExpaned’ and ‘
DataSource’. ’
DataBind’ Method will be used when we use the control with XAML page as the control will not be initialized with data source passed. ‘
FirstExpaned’ is a boolean to decide whether the control should appear with all items collapsed or first one expanded and ‘
DataSource’ Property of the control will require the list of ‘AccordionSource’ class defined as below:
- using System;
- public class AccordionSource
- {
- public string HeaderText
- {
- get;
- set;
- }
- public Color HeaderTextColor
- {
- get;
- set;
- }
- public Color HeaderBackGroundColor
- {
- get;
- set;
- }
- public View ContentItems
- {
- get;
- set;
- }
- }
The property names of the class are pretty self explanatory. The three properties containing ‘
Header’ are for the Header button created for each accordion item and the ContentItems is of View class so that we can put any container object inside it like ListView, StackView etc. In order to check whether the accordion header is expanded or not and to identify the Content associated with the button in order to show/hide, we require a button with some extra properties and we will get those by following ‘
AccordionButton’ Class:
- public class AccordionButton: Button {
- #region Private Variables
- bool mExpand = false;
- #endregion
- public AccordionButton() {
- HorizontalOptions = LayoutOptions.FillAndExpand;
- BorderColor = Color.Black;
- BorderRadius = 5;
- BorderWidth = 0;
- }#region Properties
- public bool Expand {
- get {
- return mExpand;
- }
- set {
- mExpand = value;
- }
- }
- public ContentView AssosiatedContent {
- get;
- set;
- }#endregion
- }
This completes the code of our user control which can be found in the ‘
accodion.cs’ class in example code. Now let''s see how to use this control in a sample application. In the application there is an accordion containing 3 Items. First one is a list, second is static data and third is again list.
The ‘
XamlExample’ page have used this control using following code in XAML:
- <?xml version="1.0" encoding="UTF-8"?>
- <ContentPage
- xmlns="http://xamarin.com/schemas/2014/forms"
- xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
- xmlns:ctrl="clr-namespace:AccordionEx;assembly=AccordionEx" x:Class="AccordionEx.XamlExample" Title="XAML Example">
- <ContentPage.Content>
- <ctrl:Accordion x:Name="MainOne" />
- </ContentPage.Content>
- </ContentPage>
And the following code in the code behind constructor of the page:
- public XamlExample ()
- {
- InitializeComponent ();
- MainOne.DataSource = GetSampleData ();
- MainOne.DataBind ();
- }
The ‘
CodeExample’ page has used this control directly in the code file using the following code:
- public CodeEaxmple()
- {
- Title = "Code Example";
- var vAccordionSource = GetSampleData();
- var vAccordionControl = new Accordion(vAccordionSource);
- Content = vAccordionControl;
- }
In both the example pages method ‘
GetSampleData’ is used to get the sample data to bind with the accordion user control. The code of ‘
GetSampleData’ method is as follows:
- public List < AccordionSource > GetSampleData()
- {
- var vResult = new List < AccordionSource > ();
-
- #region First List View
- var vListOne = new List < SimpleObject > ();
- for (var iCount = 0; iCount < 6; iCount++)
- {
- var vObject = new SimpleObject()
- {
- TextValue = "ObjectNo-" + iCount.ToString(),
- DataValue = iCount.ToString()
- };
- vListOne.Add(vObject);
- }
- var vListViewOne = new ListView()
- {
- ItemsSource = vListOne,
- ItemTemplate = new DataTemplate(typeof(ListDataViewCell))
- };
- vListViewOne.ItemTapped += OnListItemClicked;
-
- #endregion# region Second List
- var vListTwo = new List < SimpleObject > ();
- var vObjectRavi = new SimpleObject()
- {
- TextValue = "S Ravi Kumar",
- DataValue = "1"
- };
- vListTwo.Add(vObjectRavi);
- var vObjectFather = new SimpleObject()
- {
- TextValue = "Father",
- DataValue = "2"
- };
- vListTwo.Add(vObjectFather);
- var vObjectTrainer = new SimpleObject()
- {
- TextValue = "Trainer",
- DataValue = "2"
- };
- vListTwo.Add(vObjectTrainer);
- var vObjectConsultant = new SimpleObject()
- {
- TextValue = "Consultant",
- DataValue = "2"
- };
- vListTwo.Add(vObjectConsultant);
- var vObjectArchitect = new SimpleObject()
- {
- TextValue = "Architect",
- DataValue = "2"
- };
- vListTwo.Add(vObjectArchitect);
- var vListViewTwo = new ListView()
- {
- ItemsSource = vListTwo,
- ItemTemplate = new DataTemplate(typeof(ListDataViewCell))
- };
- vListViewTwo.ItemTapped += OnListItemClicked;
-
- #endregion
- #region StackLayout
-
- var vViewLayout = new StackLayout()
- {
- Children = {
- new Label
- {
- Text = "Static Content:"
- },
- new Label
- {
- Text = "Name : S Ravi Kumar"
- },
- new Label
- {
- Text = "Roles : Father,Trainer,Consultant,Architect"
- }
- }
- };
-
- #endregion
- var vFirstAccord = new AccordionSource()
- {
- HeaderText = "First",
- HeaderTextColor = Color.Black,
- HeaderBackGroundColor = Color.Yellow,
- ContentItems = vListViewTwo
- };
- vResult.Add(vFirstAccord);
- var vSecond = new AccordionSource()
- {
- HeaderText = "Second ",
- HeaderTextColor = Color.White,
- HeaderBackGroundColor = Color.FromHex("#77d065"),
- ContentItems = vViewLayout
- };
- vResult.Add(vSecond);
- var vThird = new AccordionSource()
- {
- HeaderText = "Third",
- HeaderTextColor = Color.White,
- HeaderBackGroundColor = Color.Purple,
- ContentItems = vListViewOne
- };
- vResult.Add(vThird);
- return vResult;
- }
In the above
GetSampleData() method as the ListView objects are created on runtime, they require a custom view cell (got from ‘
ListDataViewCell’) which shows the ‘
TextValue’ property of the ‘
SimpleObject’ class in a Label inside Stacklayout. The code of both the classes are in ‘
App.cs’ file so that they can be utilized from both the example pages.
The code of ‘
ListDataViewCell’ and ‘
SimpleObject’ class are as follows:
- public class ListDataViewCell: ViewCell
- {
- public ListDataViewCell()
- {
- var label = new Label()
- {
- Font = Font.SystemFontOfSize(NamedSize.Default),
- TextColor = Color.Blue
- };
- label.SetBinding(Label.TextProperty, new Binding("TextValue"));
- label.SetBinding(Label.ClassIdProperty, new Binding("DataValue"));
- View = new StackLayout()
- {
- Orientation = StackOrientation.Vertical,
- VerticalOptions = LayoutOptions.StartAndExpand,
- Padding = new Thickness(12, 8),
- Children = {
- label
- }
- };
- }
- }
- public class SimpleObject
- {
- public string TextValue
- {
- get;
- set;
- }
- public string DataValue
- {
- get;
- set;
- }
- }
Apart from the above code, sample application contains a ‘
HomePage’ (written in XAML as I like it more) with two buttons to display the XAML Example and CodeExample page. The code of HomePage is as follows:
XAML Code:
- <?xml version="1.0" encoding="UTF-8"?>
- <ContentPage
- xmlns="http://xamarin.com/schemas/2014/forms"
- xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" x:Class="AccordionEx.HomePage" Title="Accordion Example" >
- <ContentPage.Resources>
- <ResourceDictionary>
- <Style TargetType="Button">
- <Setter Property="BorderRadius" Value="10" />
- <Setter Property="BorderWidth" Value="2" />
- <Setter Property="WidthRequest" Value="150" />
- <Setter Property="HeightRequest" Value="150" />
- <Setter Property="HorizontalOptions" Value="Center" />
- <Setter Property="VerticalOptions" Value="Center" />
- <Setter Property="FontSize" Value="Medium" />
- <Setter Property="TextColor" Value="Red" />
- </Style>
- </ResourceDictionary>
- </ContentPage.Resources>
- <ContentPage.Content>
- <StackLayout VerticalOptions = "Center">
- <Button Text="Xaml Page" Clicked="OnXamlClicked" />
- <Button Text="Code Page" Clicked="OnCodeClicked" />
- </StackLayout>
- </ContentPage.Content>
- </ContentPage>
Code Behind C# code: - public partial class HomePage: ContentPage
- {
- public HomePage()
- {
- InitializeComponent();
- }
- public void OnXamlClicked(object sender, EventArgs args)
- {
- Navigation.PushAsync(new XamlExample());
- }
- public void OnCodeClicked(object sender, EventArgs args)
- {
- Navigation.PushAsync(new CodeEaxmple());
- }
- }
The above written ‘
HomePage’ is invoked in ‘
App.cs’ constructor using Navigation Page, the code for same is as follows:
- public App ()
- {
-
- MainPage = new NavigationPage(new HomePage());
- }
This is how the sample application looks on emulators:
Xamarin Android Player:
The source code of sample application containing accordion user control and pages using the control can be downloaded from
Github.
In the above post I have created a bare bone, accordion user control which anyone can customize as per their requirements, let me know if I have missed anything.
Read more articles on Xamarin: