Search
Tutorial 4: Dashboard Layout

Dashboard objects let you create more elaborate layouts that contain multiple plots and axes, and/or multiple kinds of chart graphics in same plot area. If you need only simple charts, use Chart -derived classes as shown in Getting Started guide and previous tutorials.

The Dashboard class represents a container of visual components such as plots, line graphics and text labels. Initially the dashboard contains two Panel classes to which its components should be added:

  • RootPanel is the root of dashboard components' hierarchy. Its elements are drawn overlapping on top of each other. Here you can add elements whose position should not depend on the size of the dashboard and their siblings. For example, Chart controls dispay LegendRenderer inside the RootPanel.
  • LayoutPanel is a vertical stack panel by default. Its components are arranged next to each other and dimensions are calculated automatically from available space. The components that we are going to create in this tutorial will be added here.

Let's start by creating a Dashboard in the main page. The dashboard can be declared in XAML or programmatically, in code. The following line creates a dashboard in XAML:

XAML  Copy Code

<mc:Dashboard x:Name="dashboard" />

Next, add a TextComponent that will display the dashboard's title. As LayoutPanel arranges its children in a vertical stack, the title will show at the top of the dashboard if it's the first component added. The following code creates and adds the title component, while also setting its appearance properties:

C#  Copy Code

dashboard.LayoutPanel.Children.Add(new TextComponent
{
    Text = "TITLE TEXT",
    FontName = "Times New Roman",
    FontStyle = FontAttributes.Bold,
    FontSize = 20.4,
    HorizontalAlignment = LayoutAlignment.Center
});

We want to divide the remaining space in two rows. For that purpose, add a GridPanel, which can host multiple components in each of its cells. Its default constructor creates one GridRow and one GridColumn, i.e. it defines a single cell. We'll add one more row to the respective collection of the grid panel:

C#  Copy Code

var grid = new GridPanel();
grid.Rows.Add(new GridRow());
dashboard.LayoutPanel.Children.Add(grid);


Now that we have defined our layout, lets create sample data series. MindFusion.Charting for .NET MAUI offers multiple classes you could use to represent data. All of them implement the Series interface, which you could implement in your own model classes if you prefer. The data class we'll use in this example is BarSeries; it takes a list of values to display as bar height, and automatically assigns bar positions at equal distances. It is convenient for creation of barcharts, but can actually be used with any other type of SeriesRenderer.

C#  Copy Code

var data1 = new BarSeries(
    new List<double>
        { 10, 15, 17, 16, 20, 15, 14, 13, 10, 17, 22, 25, 25, 25, 21, 21 },
    null, null,
    new List<string> {"nov 2014", "dec 2014",
        "01.15", "feb 2015", "mar 2015", "apr 2015",
        "05.15", "jun 2015", "jul 2015", "aug 2015",
        "09.15", "oct 2015", "nov 2015", "dec 2015",
        "01.16", "feb 2016"})
{
    Title = "LP Vinyl"
};

var data2 = new BarSeries(
 new List<double> { 13, 13, 14, 16, 23, 25, 20, 19, 16, 17, 15, 13, 13, 17, 18, 21 },
 null, null)
{
    Title = "CD"
};

var data3 = new BarSeries(
    new List<double> { 29, 28, 29, 31, 33, 35, 33, 27, 28, 28, 27, 29, 29, 31, 33, 29 },
    null, null)
{
    Title = "MP3"
};

You can see that BarSeries has two overloaded constructors. They let you specify different kinds of labels for the bars. MindFusion.Charting for .NET MAUI supports six kinds of labels displayed at various locations of rendered graphics. BarSeries lets you define inner labels drawn inside bars, top labels drawn above their bars, and X axis labels drawn along the axis. In the example above we'll only show axis labels, passing null for the other arguments. The Title values will be drawn by a LegendRenderer we are going to add later.

For visualizing the data, you must create an instance of a class derived from SeriesRenderer, each drawing a different kind of graphics. In this tutorial we'll see how several of them work, starting with a LineRenderer:

C#  Copy Code

var lineRenderer = new LineRenderer(
    new System.Collections.ObjectModel.ObservableCollection<Series> { data1, data2, data3 }
);

Apart from the kind of graphics drawn, a SeriesRenderer lets you specify styling attributes for the graphic primitives, such as stroke and fill colors:

C#  Copy Code

lineRenderer.SeriesStyle = new PerSeriesStyle
{
    Strokes = new List<MindFusion.Drawing.Brush>
    {
        new MindFusion.Drawing.SolidBrush(Colors.Red),
        new MindFusion.Drawing.SolidBrush(Colors.Blue),
        new MindFusion.Drawing.SolidBrush(Colors.Green)
    }
};

SeriesRenderer objects must be placed inside Plot components. In this tutorial we'll use three different kinds of plots: Plot2D and Plot3D for drawing in 2D and 3D Cartesian coordinate systems respectively, and RadarPlot for drawing in polar coordinate system. For the LineRenderer we defined above, create a Plot2D and set its background:

C#  Copy Code

var linePlot = new Plot2D
{
    Background = new MindFusion.Drawing.LinearGradientBrush(
        Colors.White, Color.FromRgb(224, 224, 224), 90)
};

Add the line renderer to plot's SeriesRenderers collection:

C#  Copy Code

linePlot.SeriesRenderers.Add(lineRenderer);

The data points renderered inside a Plot2D are scaled relatively to the Axis ranges defined via its XAxis and YAxis properties. For each Axis you can define its first and last visible values by setting MinValue and MaxValue properties. If their values are not specified, the control will calculate them automatically to fit all data provided through plot's SeriesRenderer objects. In adition, the Interval property lets you specify the distance between coordinate labels. The Axis class also has a Title property that lets you show description for the measure represented by an axis:

C#  Copy Code

linePlot.XAxis = new Axis();
linePlot.XAxis.MinValue = 0;
linePlot.XAxis.MaxValue = 15;
linePlot.XAxis.Interval = 1;
linePlot.XAxis.Title = "When?";

linePlot.YAxis = new Axis();
linePlot.YAxis.MinValue = 0;
linePlot.YAxis.MaxValue = 35;
linePlot.YAxis.Interval = 5;
linePlot.YAxis.Title = "How many thousands?";


If you want to show coordinates for the values drawn inside a plot, you must also add XAxisRenderer and YAxisRenderer components to the dashboard. Each of them can optionally display Axis coordinates and Title, ticks and data labels. In order to display data labels from a series, you must also set the LabelsSource property to either a SeriesRenderer or a Plot:

C#  Copy Code

var xRenderer = new XAxisRenderer(linePlot.XAxis)
{
    LabelsSource = linePlot,
    ShowCoordinates = false
};

var yRenderer = new YAxisRenderer(linePlot.YAxis);

In order to align axes to their plot, we'll add an intermediate GridPanel inside the top-left cell of the parent grid. We can do this on a couple of lines of code with the help of LayoutBuilder. That class lets us build layout fragments from several components and add them to a parent panel. In this case we'll call the CreatePlotWithBottomAndLeftAxes method to automatically create a GridPanel and add specified components to it. Other variants of this method can also automatically add the created panel to dashboard's LayoutPanel, but in this tutorial we'll add the result to our custom child panel.

C#  Copy Code

var builder = new LayoutBuilder(dashboard);
var panel1 = builder.CreatePlotWithBottomAndLeftAxes(
    linePlot, xRenderer, yRenderer);

We want to display this fragment inside top-left cell of the GridPanel we created earlier. This can be done by setting the GridRow and GridColumn properties defined in base Component class, which specify respectively the row or column index of a cell inside parent grid panel. The index is zero-based, so we set both properties to zero:

C#  Copy Code

panel1.GridRow = 0;
panel1.GridColumn = 0;
grid.Children.Add(panel1);

Following the same approach, we can display same data using a different kind of graphics and with different appearance attributes. Lets draw bars this time:

C#  Copy Code

var barDataRenderer = new BarRenderer(
    new System.Collections.ObjectModel.ObservableCollection<Series> { data1, data2, data3 }
);

We'll specify a different fill color for each series of bars by providing a list of brushes to PerSeriesStyle class.

C#  Copy Code

barDataRenderer.SeriesStyle = new PerSeriesStyle
{
    Fills = new List<MindFusion.Drawing.Brush>
    {
        new MindFusion.Drawing.SolidBrush(Color.FromRgba(Colors.Purple.Red, Colors.Purple.Green, Colors.Purple.Blue, 0.6)),
        new MindFusion.Drawing.SolidBrush(Color.FromRgba(Colors.Khaki.Red, Colors.Khaki.Green, Colors.Khaki.Blue, 0.6)),
        new MindFusion.Drawing.SolidBrush(Color.FromRgba(Colors.Orange.Red, Colors.Orange.Green, Colors.Orange.Blue, 0.6))
    }
};

You could add the BarRenderer to the same plot that now draws line graphics to create a mixed chart, but lets create a new plot for drawing bars only:

C#  Copy Code

var barPlot = new Plot2D();

Lets show axes for this plot too. Note that we are assigning same range properties as for the linechart, but through a new Axis instance. If you reused the Axis instance (by setting barPlot.XAxis = linePlot.XAxis), the plots would scroll together when the user pans any of them.

C#  Copy Code

barPlot.XAxis = new Axis();
barPlot.XAxis.MinValue = 0;
barPlot.XAxis.MaxValue = 15;
barPlot.XAxis.Interval = 1;
barPlot.XAxis.Title = "When?";

barPlot.YAxis = new Axis();
barPlot.YAxis.MinValue = 0;
barPlot.YAxis.MaxValue = 35;
barPlot.YAxis.Interval = 5;
barPlot.YAxis.Title = "How many thousands?";

Add the bar renderer to its plot:

C#  Copy Code

barPlot.SeriesRenderers.Add(barDataRenderer);


Use LayoutBuilder again to align axes to plot and add them to a grid panel:

C#  Copy Code

var panel2 = builder.CreatePlotWithBottomAndRightAxes(barPlot,
    new XAxisRenderer(barPlot.XAxis)
    {
        LabelsSource = linePlot,
        ShowCoordinates = false
    },
    new YAxisRenderer(barPlot.YAxis));

panel2.GridColumn = 1;
panel2.GridRow = 0;
grid.Children.Add(panel2);

Now create a legend and the dashboard is complete. The legend will display the series titles we assigned earlier:

C#  Copy Code

var legend = new LegendRenderer();
legend.Content.Add(lineRenderer);
dashboard.RootPanel.Children.Add(legend);

If you run the application, you should now see this image: