Why Svelte has replaced (and complemented) D3 as my go-to tool for powerful visualizations
July 2021
svelted3tutorial
When I started making data visualizations, I considered D3āData Driven Documentsāto be the gold standard programming language required to create beautiful graphics on the web.
As I progress in my career, Iām realizing that D3 has a more particular (and smaller) role in the visualization lifecycle than I initially envisioned. As I design more visualizations, Iām learning that my most efficient and intuitive development comes when I program literally.
What does that mean? (To be honest, Iām making up the term.) It means that Iām moving away from pseudo-declarative data visualization in D3, and starting to make my visualizations literally, by simply writing markup in Svelte.
Although D3 claims to be declarative, it still uses method chaining to provide instructions to render visuals. In Svelte, we donāt provide instructions but instead render our SVG elements directly, using {#each} blocks. By writing my markup literally and appending data inline, my code makes more sense and causes fewer headaches.
In this post, Iāll 1) provide an overview of D3, and how it made the process of creating visualizations so much easier; 2) explain why Iām moving away from D3 for DOM manipulation, and instead using Svelte āliterallyā; and 3) provide a funky burger š example to explain my logic.
This is not a comprehensive tutorial about how to use D3 and Svelte together. Iāll cover that in the future. In the meantime, check out tutorials from Matthias Stahl, examples of Svelte and D3 in action on The Puddingās GitHub, and an example of similar framework-driven logic on Amelia Wattenbergerās blog.
Want an immediate example? Hereās an (admittedly verbose) Svelte component I used in a recent project about Bob Ross.
D3 allows for intuitive transformations of the DOM by leveraging easy-to-understand selection syntax. Sound confusing? Letās learn through an example (adapted from the D3 homepage). Imagine we had 5 circles and wanted to change the fill of each.
In the traditional HTML DOM model, we would 1) select all circles, 2) loop through each one, and 3) redefine its fill. In code, that would look like this:
var circles = document.getElementsByTagName("circle");
for (var i = 0; i < circles.length; i++) {
var circle = circles.item(i);
circle.setAttribute("fill", "white", null);
}
js
Doing this in D3 would reduce the length of our code by a factor of 5, and allows us to write in a way that just makes sense. Here, we select every circle and change its fill.
d3.selectAll("circle").style("fill", "white")
js
Imagine if you also wanted to bind data to those circles. That is, you wanted to fill the circle according to some attribute, or size its radius according to some datapoint. In traditional JavaScript, that might look something like this:
let data = [5, 10, 15, 20, 25];
var circles = document.getElementsByTagName("circle");
for (var i = 0; i < circles.length; i++) {
var circle = circles.item(i);
circle.style.setProperty("r", data[i], null);
}
js
Not too hardābut also, not too easy. D3 simplifies this logic by removing the need for a loop and iterating on your selection for you:
let data =[5,10,15,20,25];
d3.selectAll("circle").data(data).attr("r",d=> d)
js
Again, we reduce the length of our code nearly 5x, and the code just makes sense.
All that to say, D3 is great. It works, and it works wonderfully. By removing the need to write highly imperative code that is unintuitive in nature, D3 saves developers time and allows for more powerful visualizations.
But Iām hardly using it at all recently. Increasingly, Iāve been using the JavaScript framework compiler Svelte to write SVG directly rather than tell JavaScript to write SVG. How? Letās see below.
Going back to the above circles, imagine if we could simply bind our data to our appās markup directly, without any intermediate code serving as instructions.
We can! Svelte (and Vue, React and other frameworks) allow for seamless interactions between our appās logic, data, and markup, so that we can embed data directly into our SVG elements. In this new paradigm, we could replace the set of D3 instructions from earlier with the following Svelte code:
In my view, there are three benefits to writing our code this way:
More intuitive authoring. It feels much more natural to write our SVG elements directly rather than provide D3 instructions on how to do so.
Less friction in translating D3 to the DOM. By writing SVG directly, we could copy an SVG element from the MDN docs, paste in our values, and see immediate results. Fewer handoffs results in fewer errors.
Reusability. Creating a robust and flexible <Circle />, <Bar />, or <Axis /> component permits consistent reuse within and across projects. D3 code is usually written as a series of blocks which lacks a natural structure and becomes difficult to reuse effectively.
The only downside (which is actually just an adjustment) is that this new approach requires you to learn how to write SVG. But isnāt it a good exercise to learn the anatomy of what weāre actually creating? Having knowledge of SVG elements and attributes will benefit any developer who creates visuals, no matter how they eventually do so.
Not convinced? Let me explain the logic one more time, with a tastier example:
Imagine we want to prepare a burger. We have an array of objects, each with an ingredient and ingredient-specfic instructions. We want to 1) create each item (insert it into the DOM), and 2) carry out its instructions (execute some function).
let ingredients = [
{item: "Top Bun", instruction: "Place at top of burger."},
{item: "Pickles", instruction: "Three pickles, please."},
{item: "Cheese", instruction: "Cheese is optional, but it makes the burger better."},
{item: "Patty", instruction: "Cook to your satisfaction. Optionally, add two patties."},
{item: "Tomato", instruction: "Should be the same width as bun, and thinly sliced."},
{item: "Lettuce", instruction: "Should be the same width as bun."},
{item: "Bottom Bun", instruction: "Place at bottom of burger."},
]
js
There are three ways to make this burger (at least, in our fantasy world where we make burgers via code).
Although I use the Svelte REPL to showcase these three examples, only the last one requires Svelte. The REPL is just a nice place to host (editable!) code š
The first option (old school) is to tell your app to loop through each ingredient, add it to the others, and stack the ingredients accordingly. For each ingredient in the loop, execute the burger-making according to our ingredient-specific instruction. This is how burgers would have been made, painfully, before D3.
The second option (new school) is to tell your app to read in each ingredient and instruction through D3 method chaining. We remove the need for our verbose for loop, and use the chain to give D3 a set of sequential instructions.
In our final option (new new school), we can simply append our instructions directly to the ingredient, literally. Here, we skip for loops and we skip D3 method chaining; instead, we componentize our general āburger itemā and pass each ingredient/instruction directly to that component. Only Svelte enables such burgers:
Which do you prefer? Itās much easier to make a burger by just making it, rather than giving instructions, no matter what form they come in. Frameworks allow for the construction of burgersāand visualizationsāliterally. We write our ingredients (or DOM elements) directly, and include our instructions (or data) in our markup. Now thatās a tasty burger.
Svelte allows visualization developers to write SVG directly (and avoid telling D3 what to do), while using Svelte syntax ({#each}, etc.) to avoid annoying and inefficient for loops.
Including logic directly in your markup ({#if}, etc.) removes the need for complex JavaScript/DOM interactions (more on that below).
Svelte works to complement D3, not replace it, by continuing to leverage the most powerful parts of its API: d3-scale, d3-array, d3-shape, etc.
By using Svelteās reactive declarations (the dollar signs š°), we can make certain variables āwatchā for state changes and update automatically. One huge benefit of this is that we can bind our scales to updating values such as the window width, and write minimal code to make our charts update on resize.
<script>
import windowWidth from "../stores/store.js";
import scaleLinear from 'd3-scale';
$: xScale = scaleLinear
.domain(data.map(d => d.value))
.range([0, $windowWidth])
</script>
svelte
With some other component watching and responding to resize events (such as Window.svelte), any SVG property depending on xScale will automatically update when your window resizes. You can also set the chart width itself to equal $windowWidth for a fully resizable, responsive chart.
In regular D3, we often use the ternary operator to define condition-specific attributes, like this:
// Circles are filled green if positive, red if negative
d3.selectAll('circle').style('fill',d=> d.value <0?'red':'green')
js
This is great, but what if we want to make more significant changes based on app-wide state? For example, imagine we want to show three different types of the same visualization on different screen sizes:
Desktop (over 1024px)
Tablet (520px to 1024px)
Mobile (under 520px)
In D3, we would achieve this by adding a resize event listener, providing custom breakpoints, and rendering different visuals if the updated window width were within a certain range. The complicated part would be having to render a different visual at each breakpoint.
One key difference between relying on D3 and leveraging the power of Svelte is that Svelte allows for conditional renderingdirectly in our markup, not just in our JavaScript logic. In other words, while vanilla JavaScript would approach our problem with the following:
window.addEventListener('resize',function(event){let newWidth = window.innerWidth;if(newWidth <520){// Hide tablet and desktop
document.getElementById("tablet").style.display ="none";
document.getElementById("desktop").style.display ="none";// Show mobile
document.getElementById("mobile").style.display ="block";}elseif(newWidth <1024){// Hide mobile and desktop
document.getElementById("mobile").style.display ="none";
document.getElementById("desktop").style.display ="none";// Show tablet
document.getElementById("tablet").style.display ="block";}else{// Hide mobile and tablet
document.getElementById("mobile").style.display ="none";
document.getElementById("tablet").style.display ="none";// Show desktop
document.getElementById("desktop").style.display ="block";}});