Skip to content

Instantly share code, notes, and snippets.

@gabrielflorit
Last active June 21, 2017 20:52
Show Gist options
  • Star 2 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save gabrielflorit/f822f40b7f20e019f381e3a266280bde to your computer and use it in GitHub Desktop.
Save gabrielflorit/f822f40b7f20e019f381e3a266280bde to your computer and use it in GitHub Desktop.
Lat/lon histogram
license: mit
height: 720
border: no

Lat/lon histogram

Made with blockup.

Display the source blob
Display the rendered blob
Raw
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
.buttons{margin:0 auto;text-align:center}.map svg{display:block;margin:0 auto}.map svg circle{fill:#8f092a;fill-opacity:.25;stroke:#8f092a}
var margin={top:10,right:10,bottom:10,left:10};d3.json("./boundary.topojson",function(t){var n=topojson.feature(t,t.objects.boundary),e=(d3.geoBounds(n),d3.geoCentroid(n),d3.geoAlbers()),r=d3.geoPath().projection(e),o=r.bounds(n),a=(o[1][0]-o[0][0])/(o[1][1]-o[0][1]),i=460,u=i*a,c=u-margin.left-margin.right,f=i-margin.top-margin.bottom,l=d3.select(".map svg").attrs({width:u,height:i}).append("g").attr("transform","translate("+margin.left+", "+margin.top+")");e.fitSize([c,f],n);var d=_(n.features).map("geometry").flatten().map("coordinates").flatten().sortBy("length").map(function(t){return turf.lineString(t)}).value(),s=_(d).map(function(t){return turf.lineDistance(t)}).sum(),m=1e3,g=m,p=_(d).map(function(t){var n=turf.lineDistance(t),e=Math.ceil(n*m/s),r=Math.min(g,e);g-=r;var o=n/r,a=d3.range(r).map(function(n){return turf.along(t,n*o)});return a}).flatten().map(function(t){return e(t.geometry.coordinates)}).map(function(t,n){return{index:n,x2:t[0],y2:t[1]}}).value(),y=5,x=_(p).groupBy(function(t){return Math.round(t.y2/y)*y}).map(function(t,n){return _(t).sortBy("x2").map(function(e,r){return{index:e.index,x1:c/2+4*r-4*t.length/2,y1:+n,x2:e.x2,y2:e.y2}}).value()}).flatten().value(),v=function(){var t=l.selectAll("circle").data(x,function(t){return t.index});t.enter().append("circle").attrs({cx:function(t){return t.x1},cy:function(t){return t.y1},r:1}).merge(t).transition("enter").duration(1e3).delay(function(t,n){return 10*n}).attrs({cx:function(t){return t.x2},cy:function(t){return t.y2},r:1})},h=function(){var t=l.selectAll("circle").data(x,function(t){return t.index});t.transition("exit").duration(250).delay(function(t,n){return 2*n}).attrs({cx:function(t){return t.x1},cy:function(t){return t.y1},r:1})};v(),document.querySelector("button.enter").addEventListener("click",v),document.querySelector("button.exit").addEventListener("click",h)});
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["script.js"],"names":["const","margin","top","right","bottom","left","d3","json","feature","topojson","objects","boundary","projection","geoBounds","geoCentroid","geoAlbers","path","geoPath","b","bounds","aspect","outerHeight","outerWidth","width","height","g","select","attrs","append","attr","fitSize","lineStrings","_","features","map","flatten","sortBy","d","turf","lineString","value","totalLength","lineDistance","sum","pointsCount","pointsRemaining","points","line","lineLength","upperCount","Math","ceil","linePointsCount","min","step","linePoints","range","along","geometry","coordinates","index","x2","y2","yStep","histogram","groupBy","round","key","i","x1","length","y1","enter","circles","selectAll","data","cx","cy","r","merge","transition","duration","delay","exit","document","querySelector","addEventListener"],"mappings":"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","file":"script.js","sourcesContent":["// Setup chart dimensions.\nconst margin = { top: 10, right: 10, bottom: 10, left: 10 }\n\n// Get GeoJSON.\nd3.json('./boundary.topojson', json => {\n\n\tconst feature = topojson.feature(json, json.objects.boundary)\n\n\t// Get feature's bounds, centroid, and projection.\n\tconst bounds = d3.geoBounds(feature)\n\tconst centroid = d3.geoCentroid(feature)\n\tconst projection = d3.geoAlbers()\n\n\t// Get the path.\n\tconst path = d3.geoPath().projection(projection)\n\n\t// Get the path's bounds (i.e., in pixels).\n\tconst b = path.bounds(feature)\n\n\t// Get aspect ratio.\n\tconst aspect = (b[1][0] - b[0][0]) / (b[1][1] - b[0][1])\n\n\tconst outerHeight = 460\n\tconst outerWidth = outerHeight * aspect\n\n\tconst width = outerWidth - margin.left - margin.right\n\tconst height = outerHeight - margin.top - margin.bottom\n\n\t// Prepare svg.\n\tconst g = d3.select('.map svg')\n\t\t\t.attrs({ width: outerWidth, height: outerHeight })\n\t\t.append('g')\n\t\t\t.attr('transform', `translate(${margin.left}, ${margin.top})`)\n\n\t// Fit the feature to the container's width.\n\tprojection.fitSize([width, height], feature)\n\n\t// Get the individual line strings.\n\tconst lineStrings = _(feature.features)\n\t\t.map('geometry')\n\t\t.flatten()\n\t\t.map('coordinates')\n\t\t.flatten()\n\t\t.sortBy('length')\n\t\t.map(d => turf.lineString(d))\n\t\t.value()\n\n\t// Calculate the overall line string length.\n\tconst totalLength = _(lineStrings)\n\t\t.map(d => turf.lineDistance(d))\n\t\t.sum()\n\n\t// Desired number of total points.\n\tconst pointsCount = 1000\n\tlet pointsRemaining = pointsCount\n\n\tconst points = _(lineStrings)\n\t\t.map(line => {\n\n\t\t\t// How many points will this line get?\n\n\t\t\t// First, calculate this line's length.\n\t\t\tconst lineLength = turf.lineDistance(line)\n\n\t\t\t// Next, get this line's points proportion, rounded up.\n\t\t\tconst upperCount = Math.ceil(lineLength * pointsCount / totalLength)\n\n\t\t\t// Don't get more points that are available.\n\t\t\tconst linePointsCount = Math.min(pointsRemaining, upperCount)\n\n\t\t\t// Make sure to update points remaining.\n\t\t\tpointsRemaining -= linePointsCount\n\n\t\t\t// Now that we know how many points this line will get,\n\t\t\t// calculate the distance between points - the step:\n\t\t\tconst step = lineLength / linePointsCount\n\n\t\t\tconst linePoints = d3.range(linePointsCount)\n\t\t\t\t.map(d => turf.along(line, d * step))\n\n\t\t\treturn linePoints\n\n\t\t})\n\t\t.flatten()\n\t\t.map(d => projection(d.geometry.coordinates))\n\t\t.map((d, index) => ({\n\t\t\tindex,\n\t\t\tx2: d[0],\n\t\t\ty2: d[1],\n\t\t}))\n\t\t.value()\n\n\tconst yStep = 5\n\n\tconst histogram = _(points)\n\t\t.groupBy(d => Math.round(d.y2 / yStep) * yStep)\n\t\t.map((value, key) =>\n\t\t\t_(value)\n\t\t\t\t.sortBy('x2')\n\t\t\t\t.map((d, i) => ({\n\t\t\t\t\tindex: d.index,\n\t\t\t\t\tx1: ((width / 2) + (i * 4)) - (value.length * 4)/2,\n\t\t\t\t\ty1: +key,\n\t\t\t\t\tx2: d.x2,\n\t\t\t\t\ty2: d.y2,\n\t\t\t\t}))\n\t\t\t\t.value())\n\t\t.flatten()\n\t\t.value()\n\n\t// This function adds the circles.\n\tconst enter = () => {\n\n\t\t// JOIN new data with old elements.\n\t\tconst circles = g.selectAll('circle')\n\t\t\t.data(histogram, d => d.index)\n\n\t\t// ENTER new elements present in new data.\n\t\tcircles.enter().append('circle')\n\t\t\t\t.attrs({\n\t\t\t\t\tcx: d => d.x1,\n\t\t\t\t\tcy: d => d.y1,\n\t\t\t\t\tr: 1,\n\t\t\t\t})\n\t\t\t.merge(circles)\n\t\t\t.transition('enter')\n\t\t\t\t.duration(1000)\n\t\t\t\t.delay((d, i) => i * 10)\n\t\t\t\t.attrs({\n\t\t\t\t\tcx: d => d.x2,\n\t\t\t\t\tcy: d => d.y2,\n\t\t\t\t\tr: 1,\n\t\t\t\t})\n\n\t}\n\n\t// This function removes the circles.\n\tconst exit = () => {\n\n\t\t// JOIN new data with old elements.\n\t\tconst circles = g.selectAll('circle')\n\t\t\t\t.data(histogram, d => d.index)\n\n\t\t// UPDATE old elements present in new data.\n\t\tcircles\n\t\t\t.transition('exit')\n\t\t\t\t.duration(250)\n\t\t\t\t.delay((d, i) => i * 2)\n\t\t\t\t.attrs({\n\t\t\t\t\tcx: d => d.x1,\n\t\t\t\t\tcy: d => d.y1,\n\t\t\t\t\tr: 1,\n\t\t\t\t})\n\n\t}\n\n\t// Fire the enter function on page load.\n\tenter()\n\n\t// Listen to button clicks.\n\tdocument.querySelector('button.enter').addEventListener('click', enter)\n\tdocument.querySelector('button.exit').addEventListener('click', exit)\n\n})\n"]}
<!DOCTYPE html>
<title>Lat/lon histogram</title>
<link href='dist.css' rel='stylesheet' />
<body>
<div class='map'>
<svg></svg>
</div>
<div class='buttons'>
<button class='enter'>Enter</button>
<button class='exit'>Exit</button>
</div>
<script src='https://d3js.org/d3.v4.min.js'></script>
<script src='https://d3js.org/d3-selection-multi.v1.min.js'></script>
<script src='https://d3js.org/topojson.v2.min.js'></script>
<script src='https://npmcdn.com/@turf/turf@3.10.2/turf.min.js'></script>
<script src='https://cdn.jsdelivr.net/lodash/4.17.4/lodash.min.js'></script>
<script src='dist.js'></script>
</body>
all:
rm boundary.topojson;
mapshaper -i ~/Downloads/cb_2015_us_nation_5m/cb_2015_us_nation_5m.shp name=boundary -clip bbox=-126,23,-65,50 -filter-slivers min-area=700000000 -lines -simplify dp 5% -o format=topojson boundary.topojson;
// Setup chart dimensions.
const margin = { top: 10, right: 10, bottom: 10, left: 10 }
// Get GeoJSON.
d3.json('./boundary.topojson', json => {
const feature = topojson.feature(json, json.objects.boundary)
// Get feature's bounds, centroid, and projection.
const bounds = d3.geoBounds(feature)
const centroid = d3.geoCentroid(feature)
const projection = d3.geoAlbers()
// Get the path.
const path = d3.geoPath().projection(projection)
// Get the path's bounds (i.e., in pixels).
const b = path.bounds(feature)
// Get aspect ratio.
const aspect = (b[1][0] - b[0][0]) / (b[1][1] - b[0][1])
const outerHeight = 460
const outerWidth = outerHeight * aspect
const width = outerWidth - margin.left - margin.right
const height = outerHeight - margin.top - margin.bottom
// Prepare svg.
const g = d3.select('.map svg')
.attrs({ width: outerWidth, height: outerHeight })
.append('g')
.attr('transform', `translate(${margin.left}, ${margin.top})`)
// Fit the feature to the container's width.
projection.fitSize([width, height], feature)
// Get the individual line strings.
const lineStrings = _(feature.features)
.map('geometry')
.flatten()
.map('coordinates')
.flatten()
.sortBy('length')
.map(d => turf.lineString(d))
.value()
// Calculate the overall line string length.
const totalLength = _(lineStrings)
.map(d => turf.lineDistance(d))
.sum()
// Desired number of total points.
const pointsCount = 1000
let pointsRemaining = pointsCount
const points = _(lineStrings)
.map(line => {
// How many points will this line get?
// First, calculate this line's length.
const lineLength = turf.lineDistance(line)
// Next, get this line's points proportion, rounded up.
const upperCount = Math.ceil(lineLength * pointsCount / totalLength)
// Don't get more points that are available.
const linePointsCount = Math.min(pointsRemaining, upperCount)
// Make sure to update points remaining.
pointsRemaining -= linePointsCount
// Now that we know how many points this line will get,
// calculate the distance between points - the step:
const step = lineLength / linePointsCount
const linePoints = d3.range(linePointsCount)
.map(d => turf.along(line, d * step))
return linePoints
})
.flatten()
.map(d => projection(d.geometry.coordinates))
.map((d, index) => ({
index,
x2: d[0],
y2: d[1],
}))
.value()
const yStep = 5
const histogram = _(points)
.groupBy(d => Math.round(d.y2 / yStep) * yStep)
.map((value, key) =>
_(value)
.sortBy('x2')
.map((d, i) => ({
index: d.index,
x1: ((width / 2) + (i * 4)) - (value.length * 4)/2,
y1: +key,
x2: d.x2,
y2: d.y2,
}))
.value())
.flatten()
.value()
// This function adds the circles.
const enter = () => {
// JOIN new data with old elements.
const circles = g.selectAll('circle')
.data(histogram, d => d.index)
// ENTER new elements present in new data.
circles.enter().append('circle')
.attrs({
cx: d => d.x1,
cy: d => d.y1,
r: 1,
})
.merge(circles)
.transition('enter')
.duration(1000)
.delay((d, i) => i * 10)
.attrs({
cx: d => d.x2,
cy: d => d.y2,
r: 1,
})
}
// This function removes the circles.
const exit = () => {
// JOIN new data with old elements.
const circles = g.selectAll('circle')
.data(histogram, d => d.index)
// UPDATE old elements present in new data.
circles
.transition('exit')
.duration(250)
.delay((d, i) => i * 2)
.attrs({
cx: d => d.x1,
cy: d => d.y1,
r: 1,
})
}
// Fire the enter function on page load.
enter()
// Listen to button clicks.
document.querySelector('button.enter').addEventListener('click', enter)
document.querySelector('button.exit').addEventListener('click', exit)
})
$red = #8f092a
.buttons
margin 0 auto
text-align center
.map
svg
display block
margin 0 auto
circle
fill $red
fill-opacity 0.25
stroke $red
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment