Friday, June 26, 2015

Building an Interactive SVG Pie Chart with Asp.Net MVC Razor and C#

In one of my recent web development projects I set out to redesign a basic pie chart. The previous implementation was generated by a javascript framework that did more than it was used for, and lacked some of the flexibility that we wanted. So for the rewrite I decided to simplify the code, and take an opportunity to learn more about SVG graphics.

Bear in mind as you read this that it's all still pretty "fresh" ...the code probably can (and should) be refactored. But this should serve as a good basic introduction. There are a number of good tutorials on the web for generating SVG pie charts. This post focuses on my implementation using Asp.Net MVC, C#, and Razor. WHat you'll see below is the combination of various bits of knowledge gathered from around the web and a fair amount of tinkering and troubleshooting. I think the final result looks pretty nice, so I figured it was worth sharing.

Why SVG? Because it's a vector-based image, it can scale up or down and never get pixellated, and looks good on any screen from desktop to mobile. The image data itself is very minimal, and because it can be written inline with your HTML, it doesn't require extra HTTP calls. The SVG code itself is pretty simple as well, so I don't need to rely on more intensive server-side image generation code. There are plenty of reasons why SVG is cool, but those are just a few that come to mind right now. SVG is pretty widely supported now across all major browsers except IE8, which at least for my current project I don't need to worry about.

I started out by writing a basic C# class to hold all the relevant information needed for a simple pie chart:

public class PieChart
{
public PieChart() {
this.Slices = new List<PieChartSlice>();
}
public List<PieChartSlice> Slices { get; set; }
public class PieChartSlice {
public decimal Value { get; set; }
public string DisplayValue { get; set; }
public string Label { get; set; }
public Color Color { get; set; }
}
}
view raw PieChart.cs hosted with ❤ by GitHub
Having this dedicated PieChart class allows me to generate pie charts throughout my application in a consistent way. In my MVC controller, I can build a view model for the page, and include a PieChart object. Then, in my Razor view, I can add a simple call to "DisplayFor" to generate the inline SVG, using a shared display template.

<div>
@Html.DisplayFor(viewModel => viewModel.MyPieChart)
</div
The bulk of the code lives in the shared razor template in Views/Shared/DisplayTemplates/PieChart.cshtml

To keep everything fairly self-contained, I decided to include the SVG-specific C# code right in the Razor file. While this may not adhere very well to separation of concerns, it certainly helps with manageability and making it easier to follow/understand. A fair tradeoff if you ask me. Here's the entire Razor file. I'll explain each section below:

@model Application.Models.PieChart
@functions {
double pieCenterX = 0;
double pieCenterY = 0;
double pieRadius = 50;
public string PieSlicePathDesc(double startAngleRadians, double angleRadians) {
bool longArcFlag = (angleRadians > Math.PI);
var startX = pieCenterX + (pieRadius * Math.Cos(startAngleRadians));
var startY = pieCenterY + (pieRadius * Math.Sin(startAngleRadians));
var endX = pieCenterX + (pieRadius * Math.Cos((startAngleRadians + angleRadians)));
var endY = pieCenterY + (pieRadius * Math.Sin((startAngleRadians + angleRadians)));
string pathDesc = string.Format("M{0:0.###},{1:0.###} L{2:0.###},{3:0.###} A{4:0.###},{4:0.###} 0 {5},1 {6:0.###},{7:0.###} z",
pieCenterX,
pieCenterY,
startX,
startY,
pieRadius,
(longArcFlag ? 1 : 0),
endX,
endY
);
return pathDesc;
}
}
@{
double startAngle = -(Math.PI/2); // (12-o-clock position)
}
<svg width="100%" height="200" viewBox="@(pieRadius * -1.1) @(pieRadius * -1.1) @(pieRadius * 1.1 * 2) @(pieRadius * 1.1 * 2)">
<style>
/* <![CDATA[ */
.pieChartSlice {
fill-opacity: 1;
stroke: white;
stroke-width: 1px;
stroke-linejoin: bevel; /* prevents "pointy ends" from overlapping onto other pie slices */
}
/* ]]> */
</style>
@foreach (var slice in Model.Slices)
{
var slicePercentage = (double)(slice.Value / Model.Slices.Sum(s => s.Value));
var sliceAngleRadians = slicePercentage * (2 * Math.PI);
var sliceColor = (System.Drawing.ColorTranslator.ToHtml(slice.Color));
<g>
<title>
@(string.Format("{0}: {1} ({2})", slice.Label, slice.DisplayValue, string.Format("{0:0.#}%", (100 * slicePercentage))))
</title>
@if (Model.Slices.Count == 1)
{
<circle class="pieChartSlice" cx="@pieCenterX" cy="@pieCenterY" r="@pieRadius" fill="@sliceColor" />
}
else
{
<path class="pieChartSlice" d="@PieSlicePathDesc(startAngle, sliceAngleRadians)" fill="@sliceColor" />
}
<animateTransform attributeName="transform" type="scale" from="1 1 0 0" to="1.05 1.05 0 0"
begin="mouseover" dur="0.1s" fill="freeze" repeatCount="1" />
<animateTransform attributeName="transform" type="scale" from="1.05 1.05 0 0" to="1 1 0 0"
begin="mouseout" dur="0.1s" fill="freeze" repeatCount="1" />
</g>
startAngle += sliceAngleRadians; // increment to the starting position for the next slice
}
</svg>
view raw PieChart.cshtml hosted with ❤ by GitHub

@functions

Normally I don't put complex functions in my razor views, so I didn't know the "@functions" block existed until now. For this application it was the perfect container for some simple trigonometric calculations.

PieSlicePathDesc

In SVG, we can use the path element to define complex shapes. In this case, we'll need to generate multiple "pie slices". This function does the math to calculate the x/y coordinates of each point, and outputs the path description in SVG's language. Basically the path description boils down to these instructions:

  • Start drawing at the center of our pie chart
  • Move to a point on the outside of the circle
  • Draw an arc with a given radius until we reach the next x/y coordinate
  • Draw a line back to the starting point to complete the shape

With all this logic defined here, we can easily call it in the razor markup later.

startAngle

For my pie chart, I want the first slice to start at the top of the circle. Here I set the starting angle.

svg

With most of the code out of the way, we can start generating the actual SVG tags. For a great reference to all the SVG tags and attributes, check out MDN's SVG Element Reference. The viewBox attribute defines the limits of the image using the SVG's internal coordinate system. I sized this image to have a padding around the outside equal to 10% of the radius. This comes in handy later for our hover effect.

style

Here in the SVG I embedded some CSS styling to apply a consistent style to all the pie slices, mainly to give them all a white outline.

@foreach loop

Using Razor, we can easily enumerate over the collection of PieChartSlice objects and generate the SVG paths.

g (Group) element

This is a basic SVG element to group together other SVG elements together. This allows us to combine the pie slice, as well as a tooltip and some hover animation elements.

title element

In SVG, the title element acts much like the title attribute in HTML. In most browsers this text will appear as a tooltip when the mouse hovers over it. I could have used other elements to write the text on the image, but I decided to keep it hidden to keep the image clean and uncluttered. If the user hovers over the pie slice, they can see the label, the value, and the percentage. The text is also available to screen readers and other web accessibility tools.

circle or path

In the event that our pie chart's data set only contains one item, we only need to draw a full circle, not a slice.  If we are generating multiple slices, the SVG path is used, and calls the PieSlicePathDesc method we defined earlier.

Hover animations with animateTransform

To add a little interactivity, I wanted to add a hover effect to each pie slice, where the slice would expand outward while hovering over it. To do this I added some animateTransform elements to the group containing the pie slice. There is one animation that expands from the center by 5% on mouseover, and another animation that shrinks back down on mouseout.

Finished!

My final implementation was a bit more complex. I added some code for generating the pie slice colors and some other changes specific to my project, but this should give you a basic idea of how it works.

To see the final result, below is some actual inline SVG generated by this code:

Thinking: 0:38:39 (39.3%) Writing Code: 0:34:16 (34.9%) Researching: 0:11:33 (11.8%) Refactoring: 0:09:52 (10%) Getting Distracted: 0:03:54 (4%)

No comments:

Post a Comment

Note: Only a member of this blog may post a comment.