Quantcast
Channel: Telerik Blogs
Viewing all articles
Browse latest Browse all 5210

Using a Candlestick Chart in Xamarin Forms

$
0
0
Whether or not a feature is supported in Xamarin Forms, if it's supported in the underlying native platforms, you can still make use of it. We walk through how you can utilize a Candlestick chart in your Xamarin Forms application.

In this post we will demonstrate that you can utilize financial series in the Telerik Chart for Xamarin Forms.

This type of series is not exposed in the current version of Telerik UI for Xamarin suite out of the box. However, the native platforms (Telerik UI for Android, Telerik UI for iOS and Telerik UI for Windows Universal) support such series. This means that despite the fact that the API is not unified for Xamarin Forms, you are still capable of using such visualizations in your applications with a bit of manual work. The final look should be similar: 

candlestickseries-xamarin-renderer

Requirements

You need to download the 2016.2.513 (2016 Q2) or later version of the Telerik UI for Xamarin Forms suite of controls and the latest Xamarin Forms version.

In order to control the visualization of the financial series among the different platforms, a customized RadCartesianChart renderer (one per platform) is required.
The next and actually last piece of the puzzle would be a level of abstraction representing the financial series. It should be used in the portable project of the solution. This abstraction would simply be a class deriving from the base series class included in our suite.

Creating Financial Series

This step is the easiest. In the portable project users can create a class deriving from the Telerik.XamarinForms.Chart.CartesianSeries. In order to follow the series naming convention it would be wise to call this class CandlesticskSeries since it will visualize candlesticks.

publicclassCandlestickSeries : CartesianSeries
{
}

Why Custom RadCartesianChart Renderer?

Customizing the renderer is required because this is the only place where you have access to the Telerik UI for XamarinForms (or XF for short) and the native control (Android, iOS and WinRT charts) at the same time.

The ItemsSource of the native component can be populated based on the ItemsSource of the XF control. For every newly created CandlestickSeries the respective native series will be created, populated with data and added to the native charting component.

Android Renderer

Creating a custom renderer is a straightforward operation. Simply create a new class deriving form Telerik.XamarinForms.ChartRenderer.Android.CartesianChartRenderer and you are ready to continue with the customization itself. The default renderer exposes the OnElementPropertyChanged() method which should be overridden for the purposes of our attempt. This is the entry point for the customization. 

publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
{
     protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
     {
          base.OnElementPropertyChanged(sender, e);   
     }
}

Next we need to ensure that the default renderer has done its job, initializing the XF and the native controls. This can be done by simply implementing our code after the base logic.

This base logic will initialize for us the chart itself, its axes, grid lines, behaviors and all known types of series along with the settings that are applied to those components. If there is something unknown to the RadCartesianChart, it will ignore it. This is why our custom code should take care of the custom series that will be used.

publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
{
     protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
     {
          base.OnElementPropertyChanged(sender, e);
          if(e.PropertyName.Equals("Renderer"))
          {
                
          }
     }
}

The easiest way to correctly create native series is to refer to the Telerik UI for Android documentation. According to the article you need to create two additional classes:

  • Java.Lang.Object representing a single data point
  • Customized DataPointBinding class mapping the properties of the newly created Java.Lang.Object to the properties defined in the business data object used in the portable project

For the purpose of this article we will create a CustomDataPointBinding which will use reflection to get the Java.Lang.Object through Xamarin. This will help you to fill the required data for the Android series. The implementation of the class should look like this:

publicclassCandlestickDataBinding : Com.Telerik.Widget.Chart.Engine.Databinding.DataPointBinding
{
     publicCandlestickDataBinding(stringname)
     {
          this.propertyName = name;
     }
     privateMethodInfo getMethod;
     privatestringpropertyName;
     publicoverrideJava.Lang.Object GetValue(Java.Lang.Object item)
     {
         var instance = item.GetType().GetProperty("Instance").GetValue(item);
         if(this.getMethod == null)
         {
              this.getMethod = instance.GetType().GetProperty(this.propertyName).GetGetMethod();
         }
         var value = this.getMethod.Invoke(instance, null);
         returnvalue.ToJavaObject();
     }
}

The next step is to create, populate and add a Com.Telerik.Widget.Chart.Visualization.CartesianChart.Series.Categorical.CandlestickSeries for every custom CandlestickSeries (defined earlier in the post):

namespaceChartFinancialSeries.Android
{
     publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
     {
         protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
         {
             base.OnElementPropertyChanged(sender, e);
             if(e.PropertyName.Equals("Renderer"))
             {
                 var chart = sender asRadCartesianChart;
                 if(chart != null)
                 {
                     foreach(var series inchart.Series)
                     {
                         if(series asPortable.CandlestickSeries == null
                         {
                             continue;
                         }
                             var androidSeries = newCom.Telerik.Widget.Chart.Visualization.CartesianChart.Series.Categorical.CandlestickSeries();
                             androidSeries.CategoryBinding = newCandlestickDataBinding("Category");
                             androidSeries.OpenBinding = newCandlestickDataBinding("Open");
                             androidSeries.HighBinding = newCandlestickDataBinding("High");
                             androidSeries.LowBinding = newCandlestickDataBinding("Low");
                             androidSeries.CloseBinding = newCandlestickDataBinding("Close");
                             if(series.ItemsSource != null)
                             {
                                 androidSeries.Data = newJava.Util.LinkedList(series.ItemsSource.OfType<object>().ToArray());
                             }
                             else
                             {
                                 androidSeries.Data = newJava.Util.LinkedList(newobject[] { });
                             }
                             this.Control.Series.Add(androidSeries);                           }
                    }
               }
          }
     }
}

With this, the implementation of the custom renderer is done. It now should be able to properly handle the custom CandlestickSeries. 

The last step is to register and use the customized RadCartesianChart renderer instead of the default one. This can be done in the MainActivity.cs file:

[assembly: Xamarin.Forms.ExportRenderer(typeof(Telerik.XamarinForms.Chart.RadCartesianChart), typeof(ChartFinancialSeries.Android.FinancialSeriesChartRenderer))]
namespaceChartFinancialSeries.Android
{
}

The customization of the Android project is now finished and we can now move to the next platform.

iOS Renderer

Creating a custom iOS renderer in its essence is not different from creating an Android renderer. You need to derive a custom class from the default Telerik.XamarinForms.ChartRenderer.iOS.CartesianChartRenderer. The entry point of the customization is the same—the OnElementPropertyChanged() method:

namespaceChartFinancialSeries.iOS
{
     publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
     {
         protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
         {
             base.OnElementPropertyChanged(sender, e);
             if(e.PropertyName.Equals("Renderer"))
             {
             }
         }
     }
}

In order to properly create the respective TKChartCandlestickSeries, you can refer to the Telerik UI for iOS documentation. The provided code in the article uses the TKChartFinancialDataPoint class as a data point of the candlestick series. Having this class provided in the Telerik suite means one less class to create.

You need to create one instance of this class for each candlestick defined as ItemsSource of the custom CandlestickSeries. This will allow the native charting component to visualize the financial data as expected.

publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
{
     protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
     {
          base.OnElementPropertyChanged(sender, e);
          if(e.PropertyName.Equals("Renderer"))
          {
               var chart = sender asRadCartesianChart;
               if(chart != null)
               {
                    foreach(var sereis inchart.Series)
                    {
                         if(series asPortable.CandlestickSeries == null)
                         {
                             continue;
                         }
                             var financialDataPoints = newList<TKChartFinancialDataPoint>();
                             foreach(BusinessDataObject dataPoint insereis.ItemsSource)
                             {
                                 if(dataPoint != null)
                                 {
                                    financialDataPoints.Add(TKChartFinancialDataPoint.DataPoint(
                                        newNSString(dataPoint.Category).ToNSObject(),
                                        newNSNumber(dataPoint.Open),
                                        newNSNumber(dataPoint.High),
                                        newNSNumber(dataPoint.Low),
                                        newNSNumber(dataPoint.Close)));
                                 }
                            }
                            var CandlestickSeries = newTKChartCandlestickSeries(financialDataPoints.ToArray());
                            this.Control.AddSeries(CandlestickSeries); 
                    }
               }
          }
     }
}

The issue that you can encounter is related to the required NSObject that will be used as category. In most cases the category is defined as string and converting it to NSObject does not seem to be a straightforward operation. The easiest solution is to use the ToNSObject() extension method that is defined in the Telerik.XamarinForms.Common.iOS.IOSTypeConversionExtensions class.

After having this implemented we are ready to register and use this renderer instead of the default one. This can be done in the AppDelegate class.

[assembly: Xamarin.Forms.ExportRenderer(typeof(Telerik.XamarinForms.Chart.RadCartesianChart), typeof(ChartFinancialSeries.iOS.FinancialSeriesChartRenderer))]
namespaceChartFinancialSeries.iOS
{
     [Register("AppDelegate")]
     publicpartialclassAppDelegate : global::Xamarin.Forms.Platform.iOS.FormsApplicationDelegate
     {
         publicoverrideboolFinishedLaunching(UIApplication app, NSDictionary options)
         {
               newFinancialSeriesChartRenderer();
               global::Xamarin.Forms.Forms.Init();
               Telerik.XamarinForms.Common.iOS.TelerikForms.Init();
               LoadApplication(newPortable.App());
               returnbase.FinishedLaunching(app, options);
         }
    }
}

WinRT Renderer

The approach with the custom WinRT renderer is the same.

namespaceChartFinancialSeries.WinRT
{
    publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
    {
        protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
        {
            base.OnElementPropertyChanged(sender, e);
            if(e.PropertyName.Equals("Renderer"))
            {
                
            }
        }
    }
}

The difference is how the native series are created and populated. In order to do this correctly, you can refer to the WinRT documentation. Fortunately, the type of the data required by the native CandlestickSeries is the same as the one used in our portable project—a list of objects. This means no additional type to type conversion will be needed. The only missing pieces are the bindings of the separate properties.

publicclassFinancialSeriesChartRenderer : CartesianChartRenderer
{
     protectedoverridevoidOnElementPropertyChanged(objectsender, System.ComponentModel.PropertyChangedEventArgs e)
     {
          base.OnElementPropertyChanged(sender, e);
          if(e.PropertyName.Equals("Renderer"))
          {
               var chart = sender asXF.RadCartesianChart;
               if(chart != null)
               {
                    foreach(var series inchart.Series)
                    {
                         if(series asPortable.CandlestickSeries == null)
                         {
                              continue;
                         }
                         var winSeries = newCandlestickSeries();
                         winSeries.ItemsSource = series.ItemsSource;
                         androidSeries.CategoryBinding = newCandlestickDataBinding("category");
                         androidSeries.OpenBinding = newCandlestickDataBinding("open");
                            androidSeries.HighBinding = newCandlestickDataBinding("high");
                         androidSeries.LowBinding = newCandlestickDataBinding("low");
                         androidSeries.CloseBinding = newCandlestickDataBinding("close");
                         this.Control.Series.Add(winSeries);
                    }
               }
          }
     }
}

As always, the last step is to register the customized renderer. This can be done in the MainPage.xaml.cs file 

[assembly: Xamarin.Forms.Platform.WinRT.ExportRenderer(typeof(Telerik.XamarinForms.Chart.RadCartesianChart), typeof(ChartFinancialSeries.WinRT.FinancialSeriesChartRenderer))]
namespaceChartFinancialSeries.WinRT
{
     publicsealedpartialclassMainPage
     {
          publicMainPage()
          {
               Telerik.XamarinForms.Common.WinRT.TelerikForms.Init();
               this.InitializeComponent();
               LoadApplication(newPortable.App());
          }
     }
}

Using the Customized Renderers

After the renderers are prepared we are ready to continue with setting up their use case. For simplicity we will use a ContentPage with just one RadCartesianChart visualizing only one CandlestickSeries with hardcoded data.

<?xml version="1.0"encoding="utf-8"?>
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             xmlns:telerikChart="clr-namespace:Telerik.XamarinForms.Chart;assembly=Telerik.XamarinForms.Chart"
             xmlns:local="clr-namespace:Portable"
             x:Class="Portable.StartPage">
  <telerikChart:RadCartesianChart >
    <telerikChart:RadCartesianChart.BindingContext>
      <local:ViewModel/>
    </telerikChart:RadCartesianChart.BindingContext>
    <telerikChart:RadCartesianChart.HorizontalAxis>
      <telerikChart:CategoricalAxis PlotMode="BetweenTicks"LabelFitMode="MultiLine"/>
    </telerikChart:RadCartesianChart.HorizontalAxis>
    <telerikChart:RadCartesianChart.VerticalAxis>
      <telerikChart:NumericalAxis />
    </telerikChart:RadCartesianChart.VerticalAxis>
    <telerikChart:RadCartesianChart.Series>
      <local:CandlestickSeries ItemsSource="{Binding Items}"/>
    </telerikChart:RadCartesianChart.Series>
  </telerikChart:RadCartesianChart>
</ContentPage>

The business object holding the data representing one point is defined like this:

namespacePortable
{
     publicclassBusinessDataObject
     {
          publicstringCategory { get;set;}
          publicdoubleOpen { get; set; }
          publicdoubleHigh { get; set; }
          publicdoubleLow { get; set; }
          publicdoubleClose { get; set; }
          publicBusinessDataObject(stringcategory, doubleopen, doubleclose, doublelow, doublehigh)
          {
               this.Category = category;
               this.Open = open;
               this.Close = close;
               this.Low = low;
               this.High = high;
          }
     }
}

The ViewModel holding the hardcoded business data looks like this:

namespacePortable
{
     publicclassViewModel
     {
          publicICollection<BusinessDataObject> Items { get; set; }
          publicViewModel()
          {
               this.Items = this.GetData();
          }
          privateICollection<BusinessDataObject> GetData()
          {
               var result = newObservableCollection<BusinessDataObject>();
               result.Add(newBusinessDataObject("2/29/2016", 72.75, 71.4, 71.4, 73));
               result.Add(newBusinessDataObject("2/26/2016", 71.0883, 71.52, 70.895, 71.7794));
               result.Add(newBusinessDataObject("2/25/2016", 69.68, 70.1, 69.68, 70.1));
               result.Add(newBusinessDataObject("2/24/2016", 66.5, 67.12, 65.99, 67.12));
               result.Add(newBusinessDataObject("2/23/2016", 68.26, 67.2, 66.832, 68.26));
               result.Add(newBusinessDataObject("2/22/2016", 70.391, 68.855, 68.855, 70.391));
               result.Add(newBusinessDataObject("2/19/2016", 67.6, 67.76, 67.6, 67.76));
               result.Add(newBusinessDataObject("2/18/2016", 67.57, 68.04, 67.57, 68.04));
               returnresult;
          }
     }
}

Let's Review

To sum up, in this blog post we provided guidelines on how to use CandlestickSeries in the RadCartesianChart, even though this series is not exposed in the Telerik UI for XamarinForms suite. We also took a look at the issues that are most likely to be encountered during implementation and resolved them. At the end we created a simple use case chart which utilized the customized renderers and series. 

Happy coding!


Viewing all articles
Browse latest Browse all 5210

Trending Articles