Generating Charts For Bots

System.Web.UI.DataVisualization.Charting and Bot Framework

Posted by Tom Cole on October 12, 2017

Building an interesting user experience through bots frequently requires the use of images, differing channels have varying capabilities and you can use the Channel Inspector web page to get an idea of what these are for your target platform.

One thing all the image presentations (Hero Card, Thumbnail Card, Attachment etc) have in common is that they ask for a URL to the image. In dynamic scenarios, like generating a chart image, a web facing URL to the image is not necessarily easy. You could create a one-off URL on your web app but that might have security implications and brings caching and other potential headaches. Another option is to generate a base64 string for the image and return a data url as you might in a HTML img tag i.e.

    <img src="........."/>

Since a bot built on Bot Framework is just an extended web app a quick way to generate a dynamic chart is to use System.Web.UI.DataVisualization.Charting. This can be found by adding a reference to the System.Web.DataVisualization assembly in your project.

To generate your chart you might use code similar to:

public string GetLineChart(Dictionary<DateTime, double> points, string title)
{
    // Create a series and add data points to it

    var series = new Series("Chart");
    foreach (var point in points)
    {
        series.Points.AddXY(point.Key, point.Value);
    }
    series.ChartType = SeriesChartType.Line;
    series.MarkerStyle = MarkerStyle.Circle;
    
    // Generate chart

    var chart = new Chart{Height = 800, Width = 800, Titles = { new Title(title)}};
    
    // Setup some styling

    chart.BackColor = Color.FromArgb(211, 223, 240);
    chart.BorderlineDashStyle = ChartDashStyle.Solid;
    chart.BackGradientStyle = GradientStyle.TopBottom;
    chart.BorderlineWidth = 1;
    chart.Palette = ChartColorPalette.BrightPastel;
    chart.BorderlineColor = Color.FromArgb(26, 59, 105);
    chart.RenderType = RenderType.BinaryStreaming;
    chart.BorderSkin.SkinStyle = BorderSkinStyle.Emboss;
    chart.AntiAliasing = AntiAliasingStyles.All;
    chart.TextAntiAliasingQuality = TextAntiAliasingQuality.Normal;
    
    // Add series to chart with area

    chart.Series.Add(series);
    var area = new ChartArea("Area");            
    chart.ChartAreas.Add(area);
    chart.ChartAreas[0].AxisX.LabelStyle.Format = "t";
    
    // Save it to a stream

    var imageStream = new System.IO.MemoryStream();
    chart.SaveImage(imageStream, ChartImageFormat.Png);

    // Convert stream to base64 string

    var base64 = Convert.ToBase64String(imageStream.ToArray());

    // Return base64 string prefixed with the relevant data URL parameters

    return $"data:image/png;base64,{base64}";
}

To then add this to a card in your dialog code you would do something like:

public async Task GetChart(IDialogContext context, IAwaitable<IMessageActivity> activity)
{
    // Some mechanism for retrieving data

    var data = GetData();

    // Call the chart method

    var chartDataUrl = GetLineChart(data, "Chart Title"); 
    
    var message = context.MakeMessage();    
    var card = new HeroCard
        {
            Title = "Chart",
            Subtitle = "Demo"
        };
    card.Images = new List<CardImage> {new CardImage(url: chartDataUrl};
    var attachment = new Attachment(contentType: HeroCard.ContentType, content: card);    
    message.Attachments.Add(attachment);

    await context.PostAsync(message);
    context.Wait(this.MessageReceived);
}

Or to return as an attachment, which may be preferable on some channels from a zoom perspective (an attachment can be opened full size in Teams / Skype whereas a Hero Card might be scaled down and hard to read).

public async Task GetChart(IDialogContext context, IAwaitable<IMessageActivity> activity)
{
    // Some mechanism for retrieving data

    var data = GetData();

    // Call the chart method

    var chartDataUrl = GetLineChart(data, "Chart Title"); 

    var message = context.MakeMessage();    
    var attachment = new Attachment(contentType: "image/png", contentUrl: url);
    message.Attachments.Add(attachment);

    await context.PostAsync(message);
    context.Wait(this.MessageReceived);
}