D3 Day 4

Advertisement

Your Ad could be here. I want to connect my readers to relavant ads. If you have a product targeted at developers, let's talk. [email protected]

Learn D3 in 5 days

For what we are creating in these posts d3 is way overkill and very verbose, but I need to start somewhere! These are just stepping stones into real custom visualizations that cannot be done in any other tool today. I still cannot explain how excited I am to say "I created that in d3!!!"

Todays Result

Today I will be learning about d3 scales, and adding them to the bar chart that we created yesterday. Follow along as I try to create something interesting.

today's_result

Recall Example 3 from yesterday

maybe a few days ago.... give me a break I have a lot of other priorities

In yesterdays post we created a working example of a horizontal bar chart that shows grades for a set of 5 students that are all in two classes; 'Math' and 'Science'. The chart is interactive, and will switch subjects at the press of a button.

d3 day 3 final result

Add Scales

One issue with that plot was that the scale was created by hand. In todays example we will let d3 take care of the scale for us. We will define a linear scale with an input range and an output range.


let xScale = d3.scaleLinear()
    .domain([50, 100])
    .range([0, width()]);

Then we will change the following .style method call from return (d[subject]-50 * 3) + 'px' to return xScale(d[subject]) = 'px'.


newBars.merge(bars)
    .transition()
    .style('width', function(d) {
        return xScale(d[subject]) + 'px'
    })
    .style('height', barHeight())

Keeping it dry

Note that if we had many different elements using the same scale with this code it would only exist in one place xScale and not separately in each style function. This makes our viz much more maintainable as we may see a need to change the scales in the future.

Adding Some Flair

To give this viz some simple flair, and a reason that we might want to use scales. I added a new set of buttons to allow us to change the chart size and see the viz respond. Check out the markup in the Final Markup section if your interested in that. I do want to point out that I used the d3 selectors to add the chart size classes to the chart.

The select api is very jQuery inspired, but the method chaining syntax feels very natural to me as my main data tools is pandas. The d3 methods feel very much like method chaining in python. In fact, besides the way the function is defined it reads very much like python. This feels very comfortable to me as I am always loosing track of braces and semicolons when writing javascript!


function chart4_size(size) {
    d3.select('#sizes')
        .selectAll('button')
        .classed('on', false)
    d3.select('#sizes')
        .select('.chart4-' + size + '-btn')
        .classed('on', true)
    d3.select('#chart4')
        .attr('class', 'chart ' + size)
    subject = document
        .getElementById('subjects')
        .querySelector('.on')
        .classList[0]
    render4(subject)
}

This is the css that we are using to change the size of our chart figure. Nothing fancy, just make full width or half width to show the responsiveness of our chart.


.big {
width: 100%
}
.small {
width: 50%
}

Final Result

d3 day4 final result

Final Markup

Most of the markup here is for the buttons and the callbacks. This is not really the focus of today's exercise. I have included the html here so that you can see how the buttons are tied in to the Final Script.


<div id='buttons'>
    <h3>Subject</h3>
    <div id='subjects'>
        <button class='math' onclick="render4('math')">Math</button>
        <button class='science' onclick="render4('science')">Science</button>
    </div>
    <h3>Chart Size</h3>
    <div id='sizes'>
        <button class='chart4-big-btn' onclick='chart4_size("big")')>Large</button>
        <button class='chart4-small-btn' onclick='chart4_size("small")'>Small</button>
    </div>
</div>

<div id="chart4" class='chart'></div>

Final Style


#content{
    max-width: 800px;
    margin: 0 auto;
}
.chart {
    display: block;
    padding: 10px;
    background: peachpuff;
    /* transition: all 500ms */
}

.bar {
    height: 30px;
    margin: 5px;
    background: teal;
}
.bar:hover{
    background: #444;
    }
button {
    background: rgb(240, 196, 211);
    border: none;
    font-size: 1.3rem;
    border-radius: 5px;
    padding: .2rem 1rem;
    margin-bottom: 1rem
}
.on {
    background: palevioletred;
}
.big {
width: 100%
}
.small {
width: 50%
}

Final script

Here is the final script so that you showing everything put together. Yes this is a lot of code for a bar chart without scales, click events, titles, tooltips, or anything fancy, but I need to start somewhere. d3.js is the language that builds fully custom vizualizations like no other tool today, and by doing a bit of practice now I will be ready for some serious stuff in the future.


// Setup the data
const data4 = [
    { name: 'Alice', math: 93, science: 84},
    { name: 'Bob', math: 73, science: 82 },
    { name: 'James', math: 92, science: 78},
    { name: 'Steve', math: 77, science: 93 },
    { name: 'Jordan', math: 80, science: 68 },
]

// Create some vanilla js functions to get the size of the chart
chart4 = document.getElementById('chart4')

let width = function() {
    return chart4.getBoundingClientRect().width
    }
let height = function() {
    return chart4.getBoundingClientRect().height
    }
let barHeight = function() {
    height() /  data4.length + 'px'
    }

// create a function to update the size of the chart
// Size is updated by adding a css class big or small
// Note: the  render function is called at the end to ensure the scale is re-rendered
function chart4_size(size) {
    d3.select('#sizes')
        .selectAll('button')
        .classed('on', false)
    d3.select('#sizes')
        .select('.chart4-' + size + '-btn')
        .classed('on', true)
    d3.select('#chart4')
        .attr('class', 'chart ' + size)
    subject = document
        .getElementById('subjects')
        .querySelector('.on')
        .classList[0]
    render4(subject)
}

// render the plot
// Note: I did need to bring the xScale and the width() call  into the render
// function to ensure that the scale was updated each time
function render4(subject) {

    d3.select('#subjects')
        .selectAll('button')
        .classed('on', false);

    d3.seect('#subjects')
        .select('.' + subject)
        .attr('class', subject + ' on');

    let xScale = d3.scaleLinear()
        .domain([0, 100])
        .range([50, width()]);

    const bars = d3.select('#chart4')
        .selectAll('div')
        .data(data4, function(d) {
            return d.name
        })
    const newBars = bars.enter()
        .append('div')
            .attr('class', 'bar')
            .style('width', 0)

    newBars.merge(bars)
        .transition()
        .style('width', function(d) {
            return xScale(d[subject]) + 'px'
        })
        .style('height', barHeight())
}

// create initial render and size
render4('math')
chart4_size('big')