Skip to content

Instantly share code, notes, and snippets.

@gopperman
Last active March 5, 2019 04:01
Show Gist options
  • Save gopperman/2a67b764ef28935c3854eba41985a710 to your computer and use it in GitHub Desktop.
Save gopperman/2a67b764ef28935c3854eba41985a710 to your computer and use it in GitHub Desktop.
Worcester
license: mit
height: 500
border: no

Map of my badge progress in Worcester, Massachusetts Pokemon gyms.

Made with blockup

body,svg{font-family:sans-serif}.info__title,tspan{font-weight:700}*{box-sizing:border-box}body{color:#666;font-size:1em}main{max-width:calc(125vh);margin:0 auto}.footnote{color:#999;font-style:italic;font-size:.8em}svg{width:100%;stroke-linejoin:round;stroke-linecap:round}path.feature{stroke-width:1px;stroke:#222}.background{stroke:#dedede;fill:#fafafa}.geopath{stroke:#666;stroke-width:.5px;opacity:.8}.info__title{font-size:1.6em}.info__summary{fill:#b7e1cd}.divider{fill:#222;font-weight:400}.to-do,.todo{fill:#f4c7c3}.bronze{fill:#cd7f32}.silver{fill:#999}.gold{fill:gold}.gym{background:#dedede;border:1px solid #fff;display:inline-block;float:left;position:relative;height:15px;width:15px;transition:all .2s}.gym--bronze{background:#cd7f32}.gym--silver{background:#999}.gym--gold{background:gold}.gym__info{display:none}.gym__title{font-size:1em;font-weight:700}.gym:hover{border:1px solid navy;cursor:pointer}.gym:hover .gym__info{background:#fff;border:2px solid navy;font-size:.8em;padding:5px;display:block;position:absolute;left:0;bottom:15px;width:195px;z-index:99}.gym:hover .gym__info p{margin:0}.gymlist{display:-ms-grid;display:grid;-ms-grid-columns:(minmax(15px,1fr)) [auto-fill];grid-template-columns:repeat(auto-fill,minmax(15px,1fr));grid-gap:0;width:100%}
!function(g){function n(c){if(t[c])return t[c].exports;var e=t[c]={i:c,l:!1,exports:{}};return g[c].call(e.exports,e,e.exports,n),e.l=!0,e.exports}var t={};n.m=g,n.c=t,n.i=function(g){return g},n.d=function(g,t,c){n.o(g,t)||Object.defineProperty(g,t,{configurable:!1,enumerable:!0,get:c})},n.n=function(g){var t=g&&g.__esModule?function(){return g.default}:function(){return g};return n.d(t,"a",t),t},n.o=function(g,n){return Object.prototype.hasOwnProperty.call(g,n)},n.p="",n(n.s=1)}([function(module,exports,__webpack_require__){"use strict";eval('\n\nObject.defineProperty(exports, "__esModule", {\n value: true\n});\nvar geoJson = function geoJson(gym) {\n return {\n geometry: {\n type: "Point",\n coordinates: [gym.gsx$long.$t, gym.gsx$lat.$t]\n },\n type: "Feature",\n "properties": {\n name: gym.gsx$name.$t,\n address: gym.gsx$address.$t,\n neighborhood: gym.gsx$neighborhood.$t,\n city: gym.gsx$city.$t,\n state: gym.gsx$state.$t,\n zip: gym.gsx$zip.$t,\n badge: gym.gsx$gopperman.$t\n }\n };\n};\n\nvar gymSummary = function gymSummary(data) {\n var badges = data.reduce(function (acc, item) {\n var badge = item.gsx$gopperman.$t.toLowerCase();\n var count = _.get(acc, badge, 0) + 1;\n\n acc[badge] = count;\n return acc;\n }, {});\n\n var total = data.length,\n bronze = badges.bronze || 0,\n silver = badges.silver || 0,\n gold = badges.gold || 0;\n\n return {\n total: total,\n bronze: bronze,\n silver: silver,\n gold: gold,\n goldComplete: gold / total,\n silverComplete: (silver + gold) / total || 1,\n bronzeComplete: (silver + gold + bronze) / total || 1,\n todo: badges[\'to-do\'] || 0\n };\n};\n\nvar pctForDisplay = function pctForDisplay(num) {\n return (num * 100).toFixed(1);\n};\n\nvar color = d3.scaleLinear().domain([1, 100]).interpolate(d3.interpolateHcl).range([d3.rgb("#CC211F"), d3.rgb(\'#FFD700\')]);\n\nexports.geoJson = geoJson;\nexports.color = color;\nexports.gymSummary = gymSummary;\nexports.pctForDisplay = pctForDisplay;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMC5qcyIsInNvdXJjZXMiOlsid2VicGFjazovLy91dGlsLmpzPzYzZjMiXSwic291cmNlc0NvbnRlbnQiOlsiY29uc3QgZ2VvSnNvbiA9IChneW0pID0+IHtcbiAgcmV0dXJuIHtcbiAgICBnZW9tZXRyeToge1xuICAgICAgdHlwZTogXCJQb2ludFwiLFxuICAgICAgY29vcmRpbmF0ZXM6IFtcbiAgICAgICAgZ3ltLmdzeCRsb25nLiR0LFxuICAgICAgICBneW0uZ3N4JGxhdC4kdFxuICAgICAgXVxuICAgIH0sXG4gICAgdHlwZTogXCJGZWF0dXJlXCIsXG4gICAgXCJwcm9wZXJ0aWVzXCI6IHtcbiAgICAgIG5hbWU6IGd5bS5nc3gkbmFtZS4kdCxcbiAgICAgIGFkZHJlc3M6IGd5bS5nc3gkYWRkcmVzcy4kdCxcbiAgICAgIG5laWdoYm9yaG9vZDogZ3ltLmdzeCRuZWlnaGJvcmhvb2QuJHQsXG4gICAgICBjaXR5OiBneW0uZ3N4JGNpdHkuJHQsXG4gICAgICBzdGF0ZTogZ3ltLmdzeCRzdGF0ZS4kdCxcbiAgICAgIHppcDogZ3ltLmdzeCR6aXAuJHQsXG4gICAgICBiYWRnZTogZ3ltLmdzeCRnb3BwZXJtYW4uJHRcbiAgICB9XG4gIH1cbn1cblxuY29uc3QgZ3ltU3VtbWFyeSA9IChkYXRhKSA9PiB7XG4gIGNvbnN0IGJhZGdlcyA9IGRhdGEucmVkdWNlKChhY2MsIGl0ZW0pID0+IHtcbiAgICAgIGNvbnN0IGJhZGdlID0gaXRlbS5nc3gkZ29wcGVybWFuLiR0LnRvTG93ZXJDYXNlKClcbiAgICAgIGNvbnN0IGNvdW50ID0gXy5nZXQoYWNjLCBiYWRnZSwgMCkgKyAxXG5cbiAgICAgIGFjY1tiYWRnZV0gPSBjb3VudFxuICAgICAgcmV0dXJuIGFjY1xuICB9LCB7fSlcblxuICBjb25zdCB0b3RhbCA9IGRhdGEubGVuZ3RoLFxuICAgIGJyb256ZSA9IGJhZGdlcy5icm9uemUgfHwgMCxcbiAgICBzaWx2ZXIgPSBiYWRnZXMuc2lsdmVyIHx8IDAsXG4gICAgZ29sZCA9IGJhZGdlcy5nb2xkIHx8IDBcblxuICByZXR1cm4ge1xuICAgIHRvdGFsOiB0b3RhbCxcbiAgICBicm9uemUsXG4gICAgc2lsdmVyLFxuICAgIGdvbGQsXG4gICAgZ29sZENvbXBsZXRlOiAoZ29sZCAvIHRvdGFsKSxcbiAgICBzaWx2ZXJDb21wbGV0ZTogKChzaWx2ZXIgKyBnb2xkKSAvIHRvdGFsKSB8fCAxLFxuICAgIGJyb256ZUNvbXBsZXRlOiAoKHNpbHZlciArIGdvbGQgKyBicm9uemUpIC8gdG90YWwpIHx8IDEsXG4gICAgdG9kbzogYmFkZ2VzWyd0by1kbyddIHx8IDAsXG4gIH1cbn1cblxuY29uc3QgcGN0Rm9yRGlzcGxheSA9IChudW0pID0+IHtcbiAgcmV0dXJuIChudW0gKiAxMDApLnRvRml4ZWQoMSlcbn1cblxuY29uc3QgY29sb3IgPSBkMy5zY2FsZUxpbmVhcigpLmRvbWFpbihbMSwxMDBdKVxuICAuaW50ZXJwb2xhdGUoZDMuaW50ZXJwb2xhdGVIY2wpXG4gIC5yYW5nZShbZDMucmdiKFwiI0NDMjExRlwiKSwgZDMucmdiKCcjRkZENzAwJyldKVxuXG5leHBvcnQge1xuICBnZW9Kc29uLFxuICBjb2xvcixcbiAgZ3ltU3VtbWFyeSxcbiAgcGN0Rm9yRGlzcGxheVxufVxuXG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIHV0aWwuanMiXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUZBO0FBT0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBUEE7QUFUQTtBQW1CQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUlBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQVJBO0FBVUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUlBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///0\n')},function(module,exports,__webpack_require__){"use strict";eval("\n\nvar _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };\n\nvar _util = __webpack_require__(0);\n\nvar margin = { top: 10, right: 30, bottom: 30, left: 30 },\n width = 960 - margin.left - margin.right,\n height = 500 - margin.top - margin.bottom;\n\nvar zips = ['01601', '01602', '01603', '01604', '01605', '01606', '01607', '01608', '01609', '01610', '01613', '01614', '01653', '01654', '01655'];\n\nvar footnote = document.querySelector('.footnote');\nvar centered = void 0,\n gymProgress = void 0,\n worcesterGyms = void 0;\n\nvar drawGyms = function drawGyms(zip) {\n var gyms = worcesterGyms.filter(function (gym) {\n return gym.gsx$zip.$t === zip;\n });\n\n var points = gyms.map(function (gym) {\n return (0, _util.geoJson)(gym);\n });\n\n map.selectAll('circle').data(points).enter().append('circle').attr(\"cx\", function (d) {\n return projection(d.geometry.coordinates)[0];\n }).attr(\"cy\", function (d) {\n return projection(d.geometry.coordinates)[1];\n }).attr(\"r\", \"3px\").attr('class', function (d) {\n return 'geopath ' + d.properties.badge.toLowerCase();\n }).attr('data-name', function (d) {\n return d.properties.name;\n });\n};\n\nvar updateInfo = function updateInfo() {\n var d = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : null;\n\n var zip = _.get(d, 'properties.ZCTA5CE10', 'overall');\n\n var divider = function divider() {\n infoBody.append('tspan').attr('class', 'divider').attr('dx', 8).attr('dy', 0).text('|');\n };\n\n var badgeSpan = function badgeSpan(badge) {\n var first = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;\n\n var tspan = infoBody.append('tspan').attr('class', badge).text(_.get(gymProgress, zip + '.' + badge));\n\n if (first) {\n tspan.attr('dx', 0).attr('dy', '1.2em');\n } else {\n tspan.attr('dx', 8).attr('dy', 0);\n }\n };\n\n infoTitle.text(zip !== 'overall' ? zip : 'Worcester');\n\n infoBody.selectAll('*').remove();\n\n badgeSpan('todo', true);\n divider(infoBody);\n\n badgeSpan('bronze');\n divider(infoBody);\n\n badgeSpan('silver');\n divider(infoBody);\n\n badgeSpan('gold');\n\n infoBody.append('tspan').attr('class', 'info__summary').attr('x', 20).attr('dy', '1.2em').text('t: ' + _.get(gymProgress, zip + '.total'));\n\n infoBody.append('tspan').attr('class', 'info__summary').attr('dx', 5).attr('dy', 0).text('comp: ' + (0, _util.pctForDisplay)(_.get(gymProgress, zip + '.goldComplete')) + '%');\n};\n\nvar mouseover = function mouseover(d) {\n // WHY DOESNT THIS WORK :(\n d3.select(undefined).style('fill', 'orange');\n\n if (!centered) {\n updateInfo(d);\n }\n};\n\nvar mouseout = function mouseout(d) {\n if (!centered) {\n updateInfo();\n }\n};\n\n// When clicked, zoom in\nvar click = function click(d) {\n var zip = _.get(d, 'properties.ZCTA5CE10');\n var x = void 0,\n y = void 0,\n k = void 0;\n\n map.selectAll('.geopath').remove();\n\n // Compute centroid of the selected path\n if (d && centered !== d) {\n var centroid = geoPathGenerator.centroid(d);\n x = centroid[0];\n y = centroid[1];\n k = 2;\n centered = d;\n } else {\n x = width / 2;\n y = height / 2;\n k = 1;\n centered = null;\n }\n\n // Highlight the clicked area\n map.selectAll('path').style('fill', function (d) {\n return centered && d === centered ? '#B6BFF0' : (0, _util.color)((0, _util.pctForDisplay)(_.get(gymProgress, _.get(d, 'properties.ZCTA5CE10') + '.goldComplete')));\n });\n\n var transformation = 'translate(' + width / 2 + ',' + height / 2 + ') scale(' + k + ') translate(' + -x + ',' + -y + ')';\n\n // Zoom\n map.transition().duration(750).attr('transform', transformation);\n\n if (_.get(d, 'properties', false)) {\n infoTitle.text(zip);\n }\n\n updateInfo(d);\n\n if (centered) {\n drawGyms(zip);\n }\n};\n\n// make D3 aware of the <svg> element in the HTML\nvar svg = d3.select(\"svg\");\n\n// get <svg> width and height from HTML instead of hard-coding values\nvar svgWidth = +svg.attr(\"viewBox\").split(\" \")[2],\n svgHeight = +svg.attr(\"viewBox\").split(\" \")[3];\n\n// define the map projection\nvar projection = d3.geoMercator().translate([svgWidth / 2, svgHeight / 2]).center([-71.8022934, 42.2754136]).rotate([0, 0, 0]).scale(170000);\n\nvar geoPathGenerator = d3.geoPath().projection(projection);\n\nvar background = svg.append('rect').attr('class', 'background').attr('width', '100%').attr('height', '100%').on('click', click);\n\nvar map = svg.append('g').attr('class', 'map');\n\nvar info = svg.append('g').attr('class', 'info');\n\nvar infoTitle = info.append('text').text('Worcester').attr('class', 'info__title').attr('x', 20).attr('y', 45);\n\nvar infoBody = info.append('text').attr('class', 'info__body').attr('x', 20).attr('y', 50);\n\n// Here comes the gym data\nd3.json('https://spreadsheets.google.com/feeds/list/1WDnFBw2ObCwRs3lQSk-dep3lh0jXqoSja59mXXb21as/1/public/values?alt=json').then(function (data) {\n worcesterGyms = _.sortBy(data.feed.entry.filter(function (d) {\n return d.gsx$city.$t === 'Worcester';\n }), function (o) {\n return o.gsx$neighborhood.$t;\n });\n\n gymProgress = _extends({\n overall: (0, _util.gymSummary)(worcesterGyms)\n }, zips.reduce(function (acc, zip) {\n acc[zip] = (0, _util.gymSummary)(worcesterGyms.filter(function (d) {\n return d.gsx$zip.$t === zip;\n }));\n return acc;\n }, {}));\n\n // Update the summary\n footnote.innerHTML = 'Gyms: ' + _.get(gymProgress, 'overall.total', 0) + '; Complete: ' + (0, _util.pctForDisplay)(_.get(gymProgress, 'overall.goldComplete', 0)) + '%';\n\n var gymList = d3.select('.gymlist');\n worcesterGyms.forEach(function (gym) {\n gymList.append('div').attr('class', 'gym gym--' + gym.gsx$gopperman.$t.toLowerCase()).append('div').attr('class', 'gym__info').html('\\n <p class=\"gym__title\">' + gym.gsx$name.$t + '</p>\\n <p>' + gym.gsx$neighborhood.$t + ', ' + gym.gsx$zip.$t + '</p>\\n <p><a target=\"_blank\" href=\"https://www.google.com/maps/search/?api=1&query=' + gym.gsx$lat.$t + ',' + gym.gsx$long.$t + '\">(map)</a></p>\\n ');\n });\n\n // Draw the map\n d3.json(\"https://raw.githubusercontent.com/OpenDataDE/State-zip-code-GeoJSON/master/ma_massachusetts_zip_codes_geo.min.json\").then(function (data) {\n var worcester = data.features.filter(function (d) {\n return zips.includes(d.properties.ZCTA5CE10);\n });\n\n map.selectAll('path').data(worcester).enter().append('path').attr('d', geoPathGenerator).attr('vector-effect', 'non-scaling-stroke').attr('class', 'feature').style('fill', function (d) {\n var zip = _.get(d, 'properties.ZCTA5CE10');\n return (0, _util.color)((0, _util.pctForDisplay)(_.get(gymProgress, zip + '.goldComplete')));\n }).attr('data-zip', function (d) {\n return _.get(d, 'properties.ZCTA5CE10', 'overall');\n }).on('mouseover', mouseover).on('mouseout', mouseout).on('click', click);\n\n updateInfo();\n });\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"1.js","sources":["webpack:///script.js?9a95"],"sourcesContent":["import { color, geoJson, gymSummary, pctForDisplay } from './util.js'\n\nconst margin = { top: 10, right: 30, bottom: 30, left: 30 },\n  width = 960 - margin.left - margin.right,\n  height = 500 - margin.top - margin.bottom\n\nconst zips = [\n  '01601',\n  '01602',\n  '01603',\n  '01604',\n  '01605',\n  '01606',\n  '01607',\n  '01608',\n  '01609',\n  '01610',\n  '01613',\n  '01614',\n  '01653',\n  '01654',\n  '01655'\n]\n\nconst footnote = document.querySelector('.footnote')\nlet centered, gymProgress, worcesterGyms\n\nconst drawGyms = (zip) => {\n  const gyms = worcesterGyms.filter(gym => {\n    return gym.gsx$zip.$t === zip\n  })\n\n  const points = gyms.map(gym => geoJson(gym))\n\n  map.selectAll('circle')\n    .data(points).enter()\n    .append('circle')\n    .attr(\"cx\", d => projection(d.geometry.coordinates)[0])\n    .attr(\"cy\", d => projection(d.geometry.coordinates)[1])\n    .attr(\"r\", \"3px\")\n    .attr('class', d => `geopath ${d.properties.badge.toLowerCase()}`)\n    .attr('data-name', d => d.properties.name)\n}\n\nconst updateInfo = (d = null) => {\n  const zip = _.get(d, 'properties.ZCTA5CE10', 'overall')\n\n  const divider = () => {\n    infoBody.append('tspan')\n      .attr('class', 'divider')\n      .attr('dx', 8)\n      .attr('dy', 0)\n      .text('|')\n  }\n\n  const badgeSpan = (badge, first = false) => {\n    const tspan = infoBody.append('tspan')\n      .attr('class', badge)\n      .text(_.get(gymProgress, `${zip}.${badge}`))\n\n    if (first) {\n      tspan\n        .attr('dx', 0)\n        .attr('dy', '1.2em')\n    } else {\n      tspan\n        .attr('dx', 8)\n        .attr('dy', 0)\n    }\n  }\n\n  infoTitle.text(zip !== 'overall' ? zip : 'Worcester')\n\n  infoBody\n    .selectAll('*')\n    .remove()\n\n  badgeSpan('todo', true)\n  divider(infoBody)\n\n  badgeSpan('bronze')\n  divider(infoBody)\n\n  badgeSpan('silver')\n  divider(infoBody)\n\n  badgeSpan('gold')\n\n  infoBody.append('tspan')\n    .attr('class', 'info__summary')\n    .attr('x', 20)\n    .attr('dy', '1.2em')\n    .text(`t: ${_.get(gymProgress, `${zip}.total`)}`)\n\n  infoBody.append('tspan')\n    .attr('class', 'info__summary')\n    .attr('dx', 5)\n    .attr('dy', 0)\n    .text(`comp: ${pctForDisplay(_.get(gymProgress, `${zip}.goldComplete`))}%`)\n}\n\nconst mouseover = (d) => {\n  // WHY DOESNT THIS WORK :(\n  d3.select(this)\n    .style('fill', 'orange')\n\n  if (!centered) {\n    updateInfo(d)\n  }\n}\n\nconst mouseout = (d) => {\n  if (!centered) {\n    updateInfo()\n  }\n}\n\n// When clicked, zoom in\nconst click = (d) => {\n  const zip = _.get(d, 'properties.ZCTA5CE10')\n  let x, y, k\n\n  map.selectAll('.geopath')\n    .remove()\n\n  // Compute centroid of the selected path\n  if (d && centered !== d) {\n    const centroid = geoPathGenerator.centroid(d)\n    x = centroid[0]\n    y = centroid[1]\n    k = 2\n    centered = d\n  } else {\n    x = width / 2\n    y = height / 2\n    k = 1;\n    centered = null\n  }\n\n  // Highlight the clicked area\n  map.selectAll('path')\n    .style('fill', (d) => {\n      return centered && d === centered ? '#B6BFF0' : color(pctForDisplay(_.get(gymProgress, `${_.get(d, 'properties.ZCTA5CE10')}.goldComplete`)))\n    })\n\n  const transformation = `translate(${width / 2},${height/2}) scale(${k}) translate(${-x},${-y})`;\n\n  // Zoom\n  map.transition()\n    .duration(750)\n    .attr('transform', transformation);\n\n  if (_.get(d, 'properties', false)) {\n    infoTitle.text(zip)\n  }\n\n  updateInfo(d)\n\n  if (centered) {\n    drawGyms(zip)\n  }\n}\n\n// make D3 aware of the <svg> element in the HTML\nconst svg = d3.select(\"svg\")\n\n// get <svg> width and height from HTML instead of hard-coding values\nconst svgWidth = +svg.attr(\"viewBox\").split(\" \")[2],\n  svgHeight = +svg.attr(\"viewBox\").split(\" \")[3];\n\n// define the map projection\nconst projection = d3\n  .geoMercator()\n  .translate([svgWidth / 2, svgHeight / 2])\n  .center([-71.8022934, 42.2754136])\n  .rotate([0, 0, 0])\n  .scale(170000);\n\nconst geoPathGenerator = d3.geoPath().projection(projection)\n\nconst background = svg.append('rect')\n  .attr('class', 'background')\n  .attr('width', '100%')\n  .attr('height', '100%')\n  .on('click', click)\n\nconst map = svg\n  .append('g')\n  .attr('class', 'map')\n\nconst info = svg\n  .append('g')\n  .attr('class', 'info')\n\nconst infoTitle = info\n  .append('text')\n  .text('Worcester')\n  .attr('class', 'info__title')\n  .attr('x', 20)\n  .attr('y', 45)\n\nconst infoBody = info\n  .append('text')\n  .attr('class', 'info__body')\n  .attr('x', 20)\n  .attr('y', 50)\n\n// Here comes the gym data\nd3\n  .json('https://spreadsheets.google.com/feeds/list/1WDnFBw2ObCwRs3lQSk-dep3lh0jXqoSja59mXXb21as/1/public/values?alt=json')\n  .then(data => {\n    worcesterGyms = _.sortBy(\n      data.feed.entry.filter(d => d.gsx$city.$t === 'Worcester'),\n      o => o.gsx$neighborhood.$t\n    )\n\n    gymProgress = {\n      overall: gymSummary(worcesterGyms),\n      ...zips.reduce((acc, zip) => {\n        acc[zip] = gymSummary(worcesterGyms.filter(d => d.gsx$zip.$t === zip))\n        return acc\n      }, {})\n\n    }\n\n    // Update the summary\n    footnote.innerHTML = `Gyms: ${_.get(gymProgress, 'overall.total', 0)}; Complete: ${pctForDisplay(_.get(gymProgress, 'overall.goldComplete', 0))}%`\n\n    const gymList = d3.select('.gymlist')\n    worcesterGyms.forEach(gym => {\n      gymList.append('div')\n        .attr('class', `gym gym--${gym.gsx$gopperman.$t.toLowerCase()}`)\n        .append('div')\n          .attr('class', 'gym__info')\n          .html(`\n            <p class=\"gym__title\">${gym.gsx$name.$t}</p>\n            <p>${gym.gsx$neighborhood.$t}, ${gym.gsx$zip.$t}</p>\n            <p><a target=\"_blank\" href=\"https://www.google.com/maps/search/?api=1&query=${gym.gsx$lat.$t},${gym.gsx$long.$t}\">(map)</a></p>\n          `)\n    })\n\n    // Draw the map\n    d3.json(\"https://raw.githubusercontent.com/OpenDataDE/State-zip-code-GeoJSON/master/ma_massachusetts_zip_codes_geo.min.json\")\n      .then(data => {\n        const worcester = data.features.filter(d => zips.includes(d.properties.ZCTA5CE10))\n\n        map\n          .selectAll('path')\n          .data(worcester)\n        .enter().append('path')\n          .attr('d', geoPathGenerator)\n          .attr('vector-effect', 'non-scaling-stroke')\n          .attr('class', 'feature')\n          .style('fill', (d) => {\n            const zip = _.get(d, 'properties.ZCTA5CE10')\n            return color(pctForDisplay(_.get(gymProgress, `${zip}.goldComplete`)))\n          })\n          .attr('data-zip', (d) => {\n            return _.get(d, 'properties.ZCTA5CE10', 'overall')\n          })\n          .on('mouseover', mouseover)\n          .on('mouseout', mouseout)\n          .on('click', click)\n\n          updateInfo();\n      })\n\n\n  })\n\n\n\n\n\n// WEBPACK FOOTER //\n// script.js"],"mappings":";;;;AAAA;AACA;AACA;AAAA;AAAA;AACA;AAGA;AACA;AAiBA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AAGA;AAAA;AACA;AAAA;AAEA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAKA;AACA;AACA;AAAA;AACA;AAAA;AACA;AAGA;AACA;AAGA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAKA;AAKA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AAMA;AACA;AACA;AACA;AAKA;AACA;AAGA;AACA;AAGA;AACA;AAMA;AACA;AAKA;AACA;AAGA;AACA;AAAA;AACA;AAAA;AACA;AAEA;AACA;AADA;AAGA;AAAA;AAAA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AASA;AACA;AACA;AACA;AAEA;AAAA;AAAA;AACA;AACA;AAQA;AACA;AACA;AAEA;AACA;AACA;AAIA;AACA;AAGA","sourceRoot":""}\n//# sourceURL=webpack-internal:///1\n")}]);
<!DOCTYPE html>
<head>
<title>blockup</title>
<link href='dist.css' rel='stylesheet' />
</head>
<body>
<main>
<svg viewBox="0 0 960 600"></svg>
<div class="gymlist"></div>
<div class="footnote"></div>
</main>
<script src='https://d3js.org/d3.v5.min.js'></script>
<script src='https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.11/lodash.min.js'></script>
<script src='https://unpkg.com/[email protected]/dist/topojson-client.min.js'></script>
<script src='dist.js'></script>
</body>
{
"standard": {
"globals": [
"d3"
]
}
}
import { color, geoJson, gymSummary, pctForDisplay } from './util.js'
const margin = { top: 10, right: 30, bottom: 30, left: 30 },
width = 960 - margin.left - margin.right,
height = 500 - margin.top - margin.bottom
const zips = [
'01601',
'01602',
'01603',
'01604',
'01605',
'01606',
'01607',
'01608',
'01609',
'01610',
'01613',
'01614',
'01653',
'01654',
'01655'
]
const footnote = document.querySelector('.footnote')
let centered, gymProgress, worcesterGyms
const drawGyms = (zip) => {
const gyms = worcesterGyms.filter(gym => {
return gym.gsx$zip.$t === zip
})
const points = gyms.map(gym => geoJson(gym))
map.selectAll('circle')
.data(points).enter()
.append('circle')
.attr("cx", d => projection(d.geometry.coordinates)[0])
.attr("cy", d => projection(d.geometry.coordinates)[1])
.attr("r", "3px")
.attr('class', d => `geopath ${d.properties.badge.toLowerCase()}`)
.attr('data-name', d => d.properties.name)
}
const updateInfo = (d = null) => {
const zip = _.get(d, 'properties.ZCTA5CE10', 'overall')
const divider = () => {
infoBody.append('tspan')
.attr('class', 'divider')
.attr('dx', 8)
.attr('dy', 0)
.text('|')
}
const badgeSpan = (badge, first = false) => {
const tspan = infoBody.append('tspan')
.attr('class', badge)
.text(_.get(gymProgress, `${zip}.${badge}`))
if (first) {
tspan
.attr('dx', 0)
.attr('dy', '1.2em')
} else {
tspan
.attr('dx', 8)
.attr('dy', 0)
}
}
infoTitle.text(zip !== 'overall' ? zip : 'Worcester')
infoBody
.selectAll('*')
.remove()
badgeSpan('todo', true)
divider(infoBody)
badgeSpan('bronze')
divider(infoBody)
badgeSpan('silver')
divider(infoBody)
badgeSpan('gold')
infoBody.append('tspan')
.attr('class', 'info__summary')
.attr('x', 20)
.attr('dy', '1.2em')
.text(`t: ${_.get(gymProgress, `${zip}.total`)}`)
infoBody.append('tspan')
.attr('class', 'info__summary')
.attr('dx', 5)
.attr('dy', 0)
.text(`comp: ${pctForDisplay(_.get(gymProgress, `${zip}.goldComplete`))}%`)
}
const mouseover = (d) => {
// WHY DOESNT THIS WORK :(
d3.select(this)
.style('fill', 'orange')
if (!centered) {
updateInfo(d)
}
}
const mouseout = (d) => {
if (!centered) {
updateInfo()
}
}
// When clicked, zoom in
const click = (d) => {
const zip = _.get(d, 'properties.ZCTA5CE10')
let x, y, k
map.selectAll('.geopath')
.remove()
// Compute centroid of the selected path
if (d && centered !== d) {
const centroid = geoPathGenerator.centroid(d)
x = centroid[0]
y = centroid[1]
k = 2
centered = d
} else {
x = width / 2
y = height / 2
k = 1;
centered = null
}
// Highlight the clicked area
map.selectAll('path')
.style('fill', (d) => {
return centered && d === centered ? '#B6BFF0' : color(pctForDisplay(_.get(gymProgress, `${_.get(d, 'properties.ZCTA5CE10')}.goldComplete`)))
})
const transformation = `translate(${width / 2},${height/2}) scale(${k}) translate(${-x},${-y})`;
// Zoom
map.transition()
.duration(750)
.attr('transform', transformation);
if (_.get(d, 'properties', false)) {
infoTitle.text(zip)
}
updateInfo(d)
if (centered) {
drawGyms(zip)
}
}
// make D3 aware of the <svg> element in the HTML
const svg = d3.select("svg")
// get <svg> width and height from HTML instead of hard-coding values
const svgWidth = +svg.attr("viewBox").split(" ")[2],
svgHeight = +svg.attr("viewBox").split(" ")[3];
// define the map projection
const projection = d3
.geoMercator()
.translate([svgWidth / 2, svgHeight / 2])
.center([-71.8022934, 42.2754136])
.rotate([0, 0, 0])
.scale(170000);
const geoPathGenerator = d3.geoPath().projection(projection)
const background = svg.append('rect')
.attr('class', 'background')
.attr('width', '100%')
.attr('height', '100%')
.on('click', click)
const map = svg
.append('g')
.attr('class', 'map')
const info = svg
.append('g')
.attr('class', 'info')
const infoTitle = info
.append('text')
.text('Worcester')
.attr('class', 'info__title')
.attr('x', 20)
.attr('y', 45)
const infoBody = info
.append('text')
.attr('class', 'info__body')
.attr('x', 20)
.attr('y', 50)
// Here comes the gym data
d3
.json('https://spreadsheets.google.com/feeds/list/1WDnFBw2ObCwRs3lQSk-dep3lh0jXqoSja59mXXb21as/1/public/values?alt=json')
.then(data => {
worcesterGyms = _.sortBy(
data.feed.entry.filter(d => d.gsx$city.$t === 'Worcester'),
o => o.gsx$neighborhood.$t
)
gymProgress = {
overall: gymSummary(worcesterGyms),
...zips.reduce((acc, zip) => {
acc[zip] = gymSummary(worcesterGyms.filter(d => d.gsx$zip.$t === zip))
return acc
}, {})
}
// Update the summary
footnote.innerHTML = `Gyms: ${_.get(gymProgress, 'overall.total', 0)}; Complete: ${pctForDisplay(_.get(gymProgress, 'overall.goldComplete', 0))}%`
const gymList = d3.select('.gymlist')
worcesterGyms.forEach(gym => {
gymList.append('div')
.attr('class', `gym gym--${gym.gsx$gopperman.$t.toLowerCase()}`)
.append('div')
.attr('class', 'gym__info')
.html(`
<p class="gym__title">${gym.gsx$name.$t}</p>
<p>${gym.gsx$neighborhood.$t}, ${gym.gsx$zip.$t}</p>
<p><a target="_blank" href="https://www.google.com/maps/search/?api=1&query=${gym.gsx$lat.$t},${gym.gsx$long.$t}">(map)</a></p>
`)
})
// Draw the map
d3.json("https://raw.githubusercontent.com/OpenDataDE/State-zip-code-GeoJSON/master/ma_massachusetts_zip_codes_geo.min.json")
.then(data => {
const worcester = data.features.filter(d => zips.includes(d.properties.ZCTA5CE10))
map
.selectAll('path')
.data(worcester)
.enter().append('path')
.attr('d', geoPathGenerator)
.attr('vector-effect', 'non-scaling-stroke')
.attr('class', 'feature')
.style('fill', (d) => {
const zip = _.get(d, 'properties.ZCTA5CE10')
return color(pctForDisplay(_.get(gymProgress, `${zip}.goldComplete`)))
})
.attr('data-zip', (d) => {
return _.get(d, 'properties.ZCTA5CE10', 'overall')
})
.on('mouseover', mouseover)
.on('mouseout', mouseout)
.on('click', click)
updateInfo();
})
})
*
box-sizing border-box
body
color #666
font-size 1em
font-family sans-serif
main
max-width calc(125vh)
margin 0 auto
.footnote
color #999
font-style italic
font-size .8em
svg
font-family sans-serif
width 100%
stroke-linejoin round
stroke-linecap round
path
&.feature
stroke-width 1px
stroke #222
tspan
font-weight bold
.background
stroke #dedede
fill #fafafa
.geopath
stroke #666
stroke-width .5px
opacity .8
.info
&__title
font-size 1.6em
font-weight bold
&__summary
fill #b7e1cd
.divider
fill #222
font-weight normal
.todo, .to-do
fill #f4c7c3
.bronze
fill #cd7f32
.silver
fill #999
.gold
fill gold
.gym
background #dedede
border 1px solid #fff
display inline-block
float left
position relative
height 15px
width 15px
transition all .2s
&--bronze
background #cd7f32
&--silver
background #999
&--gold
background gold
&__info
display none
&__title
font-size 1em
font-weight bold
&:hover
border 1px solid navy
cursor: pointer
.gym__info
background #fff
border 2px solid navy
font-size .8em
padding 5px
display block
position absolute
left 0
bottom 15px
width 195px
z-index 99
p
margin 0
.gymlist
display grid
grid-template-columns repeat(auto-fill, minmax(15px, 1fr))
grid-gap 0px
width 100%
const geoJson = (gym) => {
return {
geometry: {
type: "Point",
coordinates: [
gym.gsx$long.$t,
gym.gsx$lat.$t
]
},
type: "Feature",
"properties": {
name: gym.gsx$name.$t,
address: gym.gsx$address.$t,
neighborhood: gym.gsx$neighborhood.$t,
city: gym.gsx$city.$t,
state: gym.gsx$state.$t,
zip: gym.gsx$zip.$t,
badge: gym.gsx$gopperman.$t
}
}
}
const gymSummary = (data) => {
const badges = data.reduce((acc, item) => {
const badge = item.gsx$gopperman.$t.toLowerCase()
const count = _.get(acc, badge, 0) + 1
acc[badge] = count
return acc
}, {})
const total = data.length,
bronze = badges.bronze || 0,
silver = badges.silver || 0,
gold = badges.gold || 0
return {
total: total,
bronze,
silver,
gold,
goldComplete: (gold / total),
silverComplete: ((silver + gold) / total) || 1,
bronzeComplete: ((silver + gold + bronze) / total) || 1,
todo: badges['to-do'] || 0,
}
}
const pctForDisplay = (num) => {
return (num * 100).toFixed(1)
}
const color = d3.scaleLinear().domain([1,100])
.interpolate(d3.interpolateHcl)
.range([d3.rgb("#CC211F"), d3.rgb('#FFD700')])
export {
geoJson,
color,
gymSummary,
pctForDisplay
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment