Impress Your Boss with Interactive Visualizations
data:image/s3,"s3://crabby-images/c47af/c47afdb9a08bd03958486755af19034b44a03e40" alt=""
Right here
data:image/s3,"s3://crabby-images/47fe4/47fe49dd42afa6a5d17ff828e152b8a9bc728064" alt=""
Impress Your Boss with Interactive Visualizations
Impressive
Interactive Visualizations
Agenda
- Describe the talk
- Apologize for not using React (Sorry)
- "Things you need to impress your boss"
- Tools, Code Samples, and Demos
- Jupyter
- Streamlit
- SVG / d3.js
- ObservableHQ
- How to / how not to design a visualization
"... Includes Python Notebooks, d3.js, and straight React SVG. ..."
-- Tim, 2019
Things you'll need
- Story to tell
- Data
- Tooling
- "Visual intuition"
Things you'll need
- Story to tell
- Data
- Tooling
- "Visual intuition"
data:image/s3,"s3://crabby-images/23507/235077f805b106f9ee70919eecb4393194ce7ca4" alt=""
Things you'll need
- Story to tell
- Data
- Tooling
- "Visual intuition"
[
{
"name": "Calibrate the Kubernetes Cluster",
"points": 2,
"owner": "Heather",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-10",
"finishDate": "2019-07-10"
},
{
"name": "Reimplement the Cache",
"points": 5,
"owner": "Eli",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-09",
"finishDate": "2019-07-16"
},
{
"name": "Restore the Secret Store",
"points": 5,
"owner": "Heather",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-08",
"finishDate": "2019-07-10"
},
{
"name": "Calibrate the Feature Flag for B",
"points": 3,
"owner": "Heather",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-08",
"finishDate": "2019-07-08"
},
{
"name": "Address the Mongo Backend",
"points": 21,
"owner": "Eli",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-08",
"finishDate": "2019-07-31"
},
{
"name": "Fix the Azure Configuration",
"points": 2,
"owner": "Sam",
"iterationStartDate": "2019-07-08",
"workStartDate": "2019-07-08",
"finishDate": "2019-07-09"
}
]
Things you'll need
- Story to tell
- Data
- Tooling
- "Visual intuition"
Jupyter
SVG/d3.js
streamlit
ObservableHQ
data:image/s3,"s3://crabby-images/79d32/79d32b66da8de1bfbcb887fd463175e7c5557b89" alt=""
data:image/s3,"s3://crabby-images/76dfb/76dfb82b54f7f516521d7de49e4f19fd5b8e0e73" alt=""
data:image/s3,"s3://crabby-images/c820f/c820f7e52070137cbaa0600f2e752ead4c8a6414" alt=""
data:image/s3,"s3://crabby-images/ae04b/ae04bfba31f7184640d039e9110d3bd03e4bbc07" alt=""
data:image/s3,"s3://crabby-images/98e7e/98e7e85c3f7116dad1364ba8e0fe30ae5424db65" alt=""
data:image/s3,"s3://crabby-images/b3dd4/b3dd488fea6c403f814d41b2edccdc74681211c2" alt=""
Jupyter
data:image/s3,"s3://crabby-images/5ea96/5ea968ae992dc9a043edede099138e2a12c17a2d" alt=""
Time for something more impressive
Streamlit
data:image/s3,"s3://crabby-images/afea8/afea8715dc29f73531dd12ae6a314e8326a6d403" alt=""
SVG / D3.js
data:image/s3,"s3://crabby-images/48d4a/48d4a9f4c133cff05e1088c5991bad96ed43226a" alt=""
SVG?
<svg width="500" height="140">
<g>
<circle
cx="50" cy="50"
r="50"/>
</g>
</svg>
data:image/s3,"s3://crabby-images/974bf/974bf340417c56d992113160b01e9e4319f9d0e3" alt=""
var width = window.innerWidth,
height = window.innerHeight
positionStrength = 0.025;
var margin = {
top: 50,
left: 100,
bottom: 50,
right: 50
}
function updateDOM() {
let dots = svg.selectAll('circle').data(workitems);
dots.enter()
.append('circle')
.style("stroke", currentStroke)
.style("fill", currentFill)
.attr("r", 5);
dots
.attr("cx", d => d.x)
.attr("cy", d => d.y)
let xaxis = d3.axisBottom(x ? x : d3.scaleLinear())
let xAxisG = svg.selectAll('g.xaxis').data(x ? [0] : [])
xAxisG.enter()
.append('g')
.classed('xaxis', true)
.call(xaxis)
.style("opacity", 0)
.attr("transform", `translate(0,${xAxisTransform})`)
.transition()
.delay(500)
.duration(750)
.style("opacity", 1);
xAxisG.exit().remove();
}
document.getElementById("first").onclick = function() {
x = d3.scaleTime()
.domain(d3.extent(workitems, d => d.finishDate))
.range([margin.left, width - margin.right]);
y = null;
simulation
.force("charge", d3.forceManyBody().distanceMax(20).strength(-15))
.force("center", null)
.force("x", d3.forceX(d => x(d.finishDate)).strength(positionStrength))
.force("y", d3.forceY(height / 2).strength(positionStrength))
.alpha(1)
.restart();
xAxisTransform = height / 2 + 50;
svg.selectAll('g.xaxis')
.transition()
.duration(500)
.attr("transform", `translate(0, ${xAxisTransform})`);
};
var simulation = d3.forceSimulation()
.alphaDecay( 1 - (0.001**(1 / 600)) )
.force("charge", d3.forceManyBody().distanceMax(20).strength(-15))
.force("collide", d3.forceCollide().radius(5))
.force("center", d3.forceCenter( width / 2, height / 2))
.nodes(workitems)
.on("tick", updateDOM);
...
// Later, in "1"
simulation
.force("charge", d3.forceManyBody().distanceMax(20).strength(-15))
.force("center", null)
.force("x", d3.forceX(d => x(d.finishDate)).strength(positionStrength))
.force("y", d3.forceY(height / 2).strength(positionStrength))
.alpha(1)
.restart();
...
// Later, in "2"
simulation
.force("charge", null)
.force("center", null)
.force("x", d3.forceX(d => x(d.finishDate)).strength(positionStrength))
.force("y", d3.forceY(d => y(d.iterationStartDate)).strength(positionStrength))
.alpha(2)
.restart();
ObservableHQ
(Last one, I promise)
Things you'll need
- Story to tell
- Data
- Tooling
- "Visual intuition"
data:image/s3,"s3://crabby-images/f9ac3/f9ac315316a121ab59cf673d4818ec92ebe45cea" alt=""
"Graphical excellence is that which gives to the viewer the greatest number of ideas in the shortest time with the least ink in the smallest space."
data:image/s3,"s3://crabby-images/f0b87/f0b87b79a4868151b6a435af65e88f14d5155324" alt=""
data\ density = {\#\ entries \over area\ of\ display}
data:image/s3,"s3://crabby-images/ef634/ef634d5fc33f07fdc12966fa6100723574cce274" alt=""
data\ density = {\#\ entries \over area\ of\ display}
data:image/s3,"s3://crabby-images/7b025/7b0252a5c2fc5b0ebde3ddcc4306336dad441a35" alt=""
data:image/s3,"s3://crabby-images/79abc/79abcf3e77930fc91e21c19bdce7fc798ef14bfa" alt=""
data:image/s3,"s3://crabby-images/e23d6/e23d6bd5e878448262540f5de26d4c60f4c7340a" alt=""
Jupyter | Streamlit | d3.js | ObservableHQ | |
---|---|---|---|---|
Language | Python | Python | Javascript | Javascript |
Easy to Learn | ✓ | ✓ | ✖ | ✖ |
Interactive | ✖ | ✓ | ✓ | ✓ |
Custom Viz | ✖ | ✖ | ✓ | ✓ |
Run locally? | ✓ | ✓ | ✓ | ✖ |
Thank You!
Tim Garvin
@tcgarvin
Dedicated to Pete Garvin
Impress Your Boss with Interactive Visualizations
By Tim Garvin
Impress Your Boss with Interactive Visualizations
Created for CodeMash 2020.
- 465