kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
232 lines (185 loc) • 20.8 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.h3GetResolution = undefined;
var _slicedToArray2 = require('babel-runtime/helpers/slicedToArray');
var _slicedToArray3 = _interopRequireDefault(_slicedToArray2);
exports.getVertices = getVertices;
exports.getCentroid = getCentroid;
exports.idToPolygonGeo = idToPolygonGeo;
exports.getCenterHex = getCenterHex;
exports.getH3VerticeTransform = getH3VerticeTransform;
exports.distortCylinderPositions = distortCylinderPositions;
exports.getRadius = getRadius;
exports.getAngle = getAngle;
var _h3Js = require('h3-js');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
exports.h3GetResolution = _h3Js.h3GetResolution;
// get vertices should return [lon, lat]
// Copyright (c) 2018 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
function getVertices(_ref) {
var id = _ref.id;
// always reverse it
return (0, _h3Js.h3ToGeoBoundary)(id, true);
}
// get centroid should return [lon, lat]
function getCentroid(_ref2) {
var id = _ref2.id;
// always reverse it to [lng, lat]
return (0, _h3Js.h3ToGeo)(id).reverse();
}
function idToPolygonGeo(_ref3, properties) {
var object = _ref3.object;
if (!object || !object.id) {
return null;
}
var vertices = getVertices(object);
return {
geometry: {
coordinates: vertices,
type: 'LineString'
},
properties: properties
};
}
function getCenterHex(_ref4, resolution) {
var latitude = _ref4.latitude,
longitude = _ref4.longitude;
return (0, _h3Js.geoToH3)(latitude, longitude, resolution);
}
// H3 hexagon are not perfect hexagon after projection, they are slightly distorted
// Here we calculate the distortion from perfect hexagon to h3 hexagon
// A mathematica proof can be found at
// https://beta.observablehq.com/@heshan0131/h3-hexagon-shape-normalize
function getH3VerticeTransform(rawVertices, centroid) {
var vertices = revertVertices(rawVertices.map(function (vt) {
return offset(vt, centroid);
}));
var radius = getRadius(vertices[0], vertices[3]);
var angle = getAngle(vertices[0], vertices[3]);
// rotate hexagon vertices, so that v0 - v3 axis parallel with xAxis
// 2___1
// 3 / \ 0
// \___/
// 4 5
//
var rotatedVertices = vertices.map(function (vt) {
return rotate([0, 0], vt, angle);
});
// vertices of a perfect hexagon
var normalVertices = getHexagonVertices(radius);
// calculate distortion
return getDistortions(rotatedVertices, normalVertices);
}
// Vertices index based on
// https://github.com/uber/luma.gl/blob/master/modules/core/src/geometry/truncated-cone-geometry.js
function distortCylinderPositions(positions, distortions) {
var primitives = distortions.map(function (_ref5, i) {
var dr = _ref5.dr,
da = _ref5.da;
return getPtOnCircle(dr, da + Math.PI * i / 3);
});
// close it
primitives.push(primitives[0]);
// starting from the 8th vertice, repeat 4 times, only replace x(0), y(1)
return positions.map(function (v, i) {
if (i > 20 && i < 21 * 5 && i % 3 < 2) {
var row = Math.floor(i / 3);
var col = i % 3;
return primitives[row % 7][col];
}
return v;
});
}
function offset(_ref6, _ref7) {
var _ref9 = (0, _slicedToArray3.default)(_ref6, 2),
px = _ref9[0],
py = _ref9[1];
var _ref8 = (0, _slicedToArray3.default)(_ref7, 2),
x0 = _ref8[0],
y0 = _ref8[1];
return [[px - x0], [py - y0]];
}
function rotate(_ref10, _ref11, radians) {
var _ref13 = (0, _slicedToArray3.default)(_ref10, 2),
cx = _ref13[0],
cy = _ref13[1];
var _ref12 = (0, _slicedToArray3.default)(_ref11, 2),
x = _ref12[0],
y = _ref12[1];
var cos = Math.cos(radians);
var sin = Math.sin(radians);
var nx = cos * (x - cx) + sin * (y - cy) + cx;
var ny = cos * (y - cy) - sin * (x - cx) + cy;
return [nx, ny];
}
function getDistance(pt0, pt1) {
var dx = pt0[0] - pt1[0];
var dy = pt0[1] - pt1[1];
var dxy = Math.sqrt(dx * dx + dy * dy);
return dxy;
}
function getRadius(pt0, pt3) {
var dxy = getDistance(pt0, pt3);
return dxy / 2;
}
function getAngle(pt0, pt3) {
var dx = pt0[0] - pt3[0];
var dy = pt0[1] - pt3[1];
var dxy = Math.sqrt(dx * dx + dy * dy);
// Calculate angle that the perpendicular hexagon vertex axis is tilted
var angle = Math.acos(dx / dxy) * Math.sign(dy);
return angle;
}
function getPtOnCircle(radius, angle) {
return [radius * Math.cos(angle), radius * Math.sin(angle)];
}
function getHexagonVertices(r) {
var ang60 = Math.PI / 3;
var pts = [];
for (var i = 0; i < 6; i++) {
pts.push(getPtOnCircle(r, ang60 * i));
}
return pts;
}
function revertVertices(verts) {
// reverting verts from clock (h3) to counter clock wise (luma cylinder)
var seq = [0, 5, 4, 3, 2, 1];
return seq.map(function (s) {
return verts[s];
});
}
function getDistortions(vts, origs) {
// 0 and 3 should be the guide
var ct = [0, 0];
var distortions = [];
for (var i = 0; i < 6; i++) {
var vt = vts[i];
var org = origs[i];
var r = getRadius(org, ct);
var dr = getRadius(vt, ct) / r;
var da = Math.atan2(vt[1], vt[0]) - Math.atan2(org[1], org[0]);
distortions.push({ dr: dr, da: da });
}
return distortions;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/h3-hexagon-layer/h3-utils.js"],"names":["getVertices","getCentroid","idToPolygonGeo","getCenterHex","getH3VerticeTransform","distortCylinderPositions","getRadius","getAngle","h3GetResolution","id","reverse","properties","object","vertices","geometry","coordinates","type","resolution","latitude","longitude","rawVertices","centroid","revertVertices","map","offset","vt","radius","angle","rotatedVertices","rotate","normalVertices","getHexagonVertices","getDistortions","positions","distortions","primitives","i","dr","da","getPtOnCircle","Math","PI","push","v","row","floor","col","px","py","x0","y0","radians","cx","cy","x","y","cos","sin","nx","ny","getDistance","pt0","pt1","dx","dy","dxy","sqrt","pt3","acos","sign","r","ang60","pts","verts","seq","s","vts","origs","ct","org","atan2"],"mappings":"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","file":"h3-utils.js","sourcesContent":["// Copyright (c) 2018 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport {h3GetResolution, h3ToGeo, h3ToGeoBoundary, geoToH3} from 'h3-js';\nexport {h3GetResolution};\n\n// get vertices should return [lon, lat]\nexport function getVertices({id}) {\n  // always reverse it\n  return h3ToGeoBoundary(id, true);\n}\n\n// get centroid should return [lon, lat]\nexport function getCentroid({id}) {\n  // always reverse it to [lng, lat]\n  return h3ToGeo(id).reverse();\n}\n\nexport function idToPolygonGeo({object}, properties) {\n  if (!object || !object.id) {\n    return null;\n  }\n\n  const vertices = getVertices(object);\n\n  return {\n    geometry: {\n      coordinates: vertices,\n      type: 'LineString'\n    },\n    properties\n  };\n}\n\nexport function getCenterHex({latitude, longitude}, resolution) {\n  return geoToH3(latitude, longitude, resolution);\n}\n\n// H3 hexagon are not perfect hexagon after projection, they are slightly distorted\n// Here we calculate the distortion from perfect hexagon to h3 hexagon\n// A mathematica proof can be found at\n// https://beta.observablehq.com/@heshan0131/h3-hexagon-shape-normalize\nexport function getH3VerticeTransform(rawVertices, centroid) {\n  const vertices = revertVertices(rawVertices.map(vt => offset(vt, centroid)));\n  const radius = getRadius(vertices[0], vertices[3]);\n\n  const angle = getAngle(vertices[0], vertices[3])\n\n  // rotate hexagon vertices, so that v0 - v3 axis parallel with xAxis\n  //   2___1\n  // 3 /   \\ 0\n  //   \\___/\n  //   4   5\n  //\n  const rotatedVertices = vertices.map(vt => rotate([0, 0], vt, angle));\n\n  // vertices of a perfect hexagon\n  const normalVertices = getHexagonVertices(radius);\n\n  // calculate distortion\n  return getDistortions(rotatedVertices, normalVertices)\n}\n\n// Vertices index based on\n// https://github.com/uber/luma.gl/blob/master/modules/core/src/geometry/truncated-cone-geometry.js\nexport function distortCylinderPositions(positions, distortions) {\n\n  const primitives = distortions.map(({dr, da}, i) =>\n    getPtOnCircle(dr, da + Math.PI * i / 3));\n  // close it\n  primitives.push(primitives[0]);\n\n  // starting from the 8th vertice, repeat 4 times, only replace x(0), y(1)\n  return positions.map((v, i) => {\n    if (i > 20 && i < 21 * 5 && i % 3 < 2) {\n      const row = Math.floor(i / 3);\n      const col = i % 3;\n      return primitives[row % 7][col];\n    }\n    return v;\n  });\n}\n\nfunction offset([px, py], [x0, y0]) {\n  return [[px - x0], [py - y0]];\n}\n\nfunction rotate([cx, cy], [x, y], radians) {\n  const cos = Math.cos(radians);\n  const sin = Math.sin(radians);\n  const nx = (cos * (x - cx)) + (sin * (y - cy)) + cx;\n  const ny = (cos * (y - cy)) - (sin * (x - cx)) + cy;\n\n  return [nx, ny];\n}\n\nfunction getDistance(pt0, pt1) {\n  const dx = pt0[0] - pt1[0];\n  const dy = pt0[1] - pt1[1];\n  const dxy = Math.sqrt(dx * dx + dy * dy);\n  return dxy;\n}\n\nexport function getRadius(pt0, pt3) {\n  const dxy = getDistance(pt0, pt3);\n  return dxy / 2;\n}\n\nexport function getAngle(pt0, pt3) {\n  const dx = pt0[0] - pt3[0];\n  const dy = pt0[1] - pt3[1];\n  const dxy = Math.sqrt(dx * dx + dy * dy);\n\n  // Calculate angle that the perpendicular hexagon vertex axis is tilted\n  const angle = Math.acos(dx / dxy) * Math.sign(dy);\n  return angle;\n}\n\nfunction getPtOnCircle(radius, angle) {\n  return [radius *  Math.cos(angle), radius *  Math.sin(angle)];\n}\n\nfunction getHexagonVertices(r) {\n  const ang60 = Math.PI / 3;\n  const pts = []\n  for (let i = 0; i < 6; i++) {\n    pts.push(getPtOnCircle(r, ang60 * i))\n  }\n\n  return pts;\n}\n\nfunction revertVertices(verts) {\n  // reverting verts from clock (h3) to counter clock wise (luma cylinder)\n  const seq = [0, 5, 4, 3, 2, 1];\n  return seq.map(s => verts[s]);\n}\n\nfunction getDistortions(vts, origs) {\n  // 0 and 3 should be the guide\n  const ct = [0, 0];\n  const distortions = [];\n\n  for (let i = 0; i < 6; i++) {\n    const vt = vts[i];\n    const org = origs[i];\n\n    const r = getRadius(org, ct);\n    const dr = getRadius(vt, ct) / r\n\n    const da = Math.atan2(vt[1], vt[0]) - Math.atan2(org[1], org[0]);\n\n    distortions.push({dr, da});\n  }\n\n  return distortions;\n}\n"]}
;