Public
Edited
Jun 9, 2023
Insert cell
Insert cell
// load the data from attached file
data = FileAttachment("gd_projects_dates.csv").csv()
Insert cell
// convert date strings to date time objects
data_dt = data.map(x => {x.date = new Date(x.date); return x});
Insert cell
degree_per_month = 360/12;
Insert cell
function toRadians(degrees) {
return (Math.PI/180) * degrees; // convert degrees to radians
}
Insert cell
data.filter(x => x.project == "lenape-timetree");
Insert cell
data.filter(x => x.project == "lenape-timetree").map(x => [new Date(x.date).getMonth(), x.issue]);
Insert cell
data.filter(x => x.project == "lenape-timetree").map(x => [new Date(x.date).getMonth() * degree_per_month, x.issue]);
Insert cell
Math.max(...data.map(x => x.issue))
Insert cell
// determine earliest year in our dataset
minYear = Math.min(...data_dt.map(x => x.date.getFullYear()))
Insert cell
// create a d3 scale to map issues to petal size
petalScale = d3.scaleLinear([0, Math.max(...data.map(x => x.issue))], [15, 350]);
Insert cell
petal = {
let petal = data_dt.filter(x => x.project == "lenape-timetree").map(x => [toRadians(x.date.getMonth() * degree_per_month), petalScale(x.issue)]);
petal.push(...data_dt.reverse().map(x => [toRadians(x.date.getMonth()), petalScale(0)]));
return petal;
}
Insert cell
{
let preDate = new Date(data_dt[0].date.getTime());
preDate.setMonth(data_dt[0].date.getMonth() - 1);
return preDate;
}
Insert cell
function monthToAngle(month) {
// our angles are zero-based but months are not, so adjust
return toRadians((month - 1) * degree_per_month)
}
Insert cell
function yearOffset(year) {
return (year - minYear) * 10;
}
Insert cell
yearOffset(2019) + petalScale(1)
Insert cell
function generatePetal(project,year) {
let project_data = data_dt.filter(x => x.project == project);
if (year != undefined) {
project_data = project_data.filter(x => x.date.getFullYear() == year);
}
// add zero counts to the month before and after the data, to make it look more like a petal
let monthOffset = (24*60*60*1000) * 31; // 31 days - we don't care where in the month it is, just previous month
let preDate = new Date(project_data[0].date.getTime() - monthOffset);
let afterDate = new Date(project_data[project_data.length - 1].date.getTime() + monthOffset);
project_data.unshift({date: preDate, issue: 0});
project_data.push({date: afterDate, issue: 0});
let petal = project_data.map(x => [
monthToAngle(x.date.getMonth()),
yearOffset(x.date.getFullYear()) + petalScale(x.issue)
]);
// petal.push(...project_data.reverse().map(x => [x[0], petalScale(0)]));
petal.push(...project_data.reverse().map(x => [
monthToAngle(x.date.getMonth()),
yearOffset(x.date.getFullYear()) + petalScale(0)]));
return petal;
}
Insert cell
data_dt.filter(x => x.project == "mep-django").filter(x => x.date.getFullYear() == 2019)
Insert cell
generatePetal("mep-django", 2019)
Insert cell
data_dt.filter(x => x.project == "cdh-web")
Insert cell
generatePetal("cdh-web")
Insert cell
svg`<svg width=800 height=700>
<g transform="translate(350,350)">
<circle cx="0" cy="0" r="14" stroke="gray" fill="none"/>
<!--<path d="${d3.lineRadial().curve(d3.curveNatural)(petal)}" fill="#61514b" stroke="#4b413e" opacity="0.5" stroke-width="2"> -->
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("lenape-timetree"))}" fill="#61514b" stroke="#4b413e" opacity="0.5" stroke-width="5"/>
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("cdh-web", 2021))}" fill="#00edff" stroke="#00edff" opacity="0.5" stroke-width="5"/>
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("mep-django", 2019))}" fill="#07ae79" stroke="#037c55" opacity="0.5" stroke-width="5"/>
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("geniza"))}" fill="#c37f97" stroke="#3d3d3d" opacity="0.5" stroke-width="5"/>
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("ppa-django"))}" fill="#f05b69" stroke="#6252a0" opacity="0.5" stroke-width="5"/>
<path d="${d3.lineRadial().curve(d3.curveNatural)(generatePetal("startwords"))}" fill="#AE80FF" stroke="#3D206C" opacity="0.5" stroke-width="5"/>
</g>
</svg>`
Insert cell

One platform to build and deploy the best data apps

Experiment and prototype by building visualizations in live JavaScript notebooks. Collaborate with your team and decide which concepts to build out.
Use Observable Framework to build data apps locally. Use data loaders to build in any language or library, including Python, SQL, and R.
Seamlessly deploy to Observable. Test before you ship, use automatic deploy-on-commit, and ensure your projects are always up-to-date.
Learn more