UNPKG

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
'use strict'; 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":";;;;;;;;;;;QAwBgBA,W,GAAAA,W;QAMAC,W,GAAAA,W;QAKAC,c,GAAAA,c;QAgBAC,Y,GAAAA,Y;QAQAC,qB,GAAAA,qB;QAuBAC,wB,GAAAA,wB;QAsCAC,S,GAAAA,S;QAKAC,Q,GAAAA,Q;;AAzGhB;;;;QACQC,e,GAAAA,qB;;AAER;AAvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAMO,SAASR,WAAT,OAA2B;AAAA,MAALS,EAAK,QAALA,EAAK;;AAChC;AACA,SAAO,2BAAgBA,EAAhB,EAAoB,IAApB,CAAP;AACD;;AAED;AACO,SAASR,WAAT,QAA2B;AAAA,MAALQ,EAAK,SAALA,EAAK;;AAChC;AACA,SAAO,mBAAQA,EAAR,EAAYC,OAAZ,EAAP;AACD;;AAEM,SAASR,cAAT,QAAkCS,UAAlC,EAA8C;AAAA,MAArBC,MAAqB,SAArBA,MAAqB;;AACnD,MAAI,CAACA,MAAD,IAAW,CAACA,OAAOH,EAAvB,EAA2B;AACzB,WAAO,IAAP;AACD;;AAED,MAAMI,WAAWb,YAAYY,MAAZ,CAAjB;;AAEA,SAAO;AACLE,cAAU;AACRC,mBAAaF,QADL;AAERG,YAAM;AAFE,KADL;AAKLL;AALK,GAAP;AAOD;;AAEM,SAASR,YAAT,QAA6Cc,UAA7C,EAAyD;AAAA,MAAlCC,QAAkC,SAAlCA,QAAkC;AAAA,MAAxBC,SAAwB,SAAxBA,SAAwB;;AAC9D,SAAO,mBAAQD,QAAR,EAAkBC,SAAlB,EAA6BF,UAA7B,CAAP;AACD;;AAED;AACA;AACA;AACA;AACO,SAASb,qBAAT,CAA+BgB,WAA/B,EAA4CC,QAA5C,EAAsD;AAC3D,MAAMR,WAAWS,eAAeF,YAAYG,GAAZ,CAAgB;AAAA,WAAMC,OAAOC,EAAP,EAAWJ,QAAX,CAAN;AAAA,GAAhB,CAAf,CAAjB;AACA,MAAMK,SAASpB,UAAUO,SAAS,CAAT,CAAV,EAAuBA,SAAS,CAAT,CAAvB,CAAf;;AAEA,MAAMc,QAAQpB,SAASM,SAAS,CAAT,CAAT,EAAsBA,SAAS,CAAT,CAAtB,CAAd;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMe,kBAAkBf,SAASU,GAAT,CAAa;AAAA,WAAMM,OAAO,CAAC,CAAD,EAAI,CAAJ,CAAP,EAAeJ,EAAf,EAAmBE,KAAnB,CAAN;AAAA,GAAb,CAAxB;;AAEA;AACA,MAAMG,iBAAiBC,mBAAmBL,MAAnB,CAAvB;;AAEA;AACA,SAAOM,eAAeJ,eAAf,EAAgCE,cAAhC,CAAP;AACD;;AAED;AACA;AACO,SAASzB,wBAAT,CAAkC4B,SAAlC,EAA6CC,WAA7C,EAA0D;;AAE/D,MAAMC,aAAaD,YAAYX,GAAZ,CAAgB,iBAAWa,CAAX;AAAA,QAAEC,EAAF,SAAEA,EAAF;AAAA,QAAMC,EAAN,SAAMA,EAAN;AAAA,WACjCC,cAAcF,EAAd,EAAkBC,KAAKE,KAAKC,EAAL,GAAUL,CAAV,GAAc,CAArC,CADiC;AAAA,GAAhB,CAAnB;AAEA;AACAD,aAAWO,IAAX,CAAgBP,WAAW,CAAX,CAAhB;;AAEA;AACA,SAAOF,UAAUV,GAAV,CAAc,UAACoB,CAAD,EAAIP,CAAJ,EAAU;AAC7B,QAAIA,IAAI,EAAJ,IAAUA,IAAI,KAAK,CAAnB,IAAwBA,IAAI,CAAJ,GAAQ,CAApC,EAAuC;AACrC,UAAMQ,MAAMJ,KAAKK,KAAL,CAAWT,IAAI,CAAf,CAAZ;AACA,UAAMU,MAAMV,IAAI,CAAhB;AACA,aAAOD,WAAWS,MAAM,CAAjB,EAAoBE,GAApB,CAAP;AACD;AACD,WAAOH,CAAP;AACD,GAPM,CAAP;AAQD;;AAED,SAASnB,MAAT,eAAoC;AAAA;AAAA,MAAnBuB,EAAmB;AAAA,MAAfC,EAAe;;AAAA;AAAA,MAATC,EAAS;AAAA,MAALC,EAAK;;AAClC,SAAO,CAAC,CAACH,KAAKE,EAAN,CAAD,EAAY,CAACD,KAAKE,EAAN,CAAZ,CAAP;AACD;;AAED,SAASrB,MAAT,iBAAkCsB,OAAlC,EAA2C;AAAA;AAAA,MAA1BC,EAA0B;AAAA,MAAtBC,EAAsB;;AAAA;AAAA,MAAhBC,CAAgB;AAAA,MAAbC,CAAa;;AACzC,MAAMC,MAAMhB,KAAKgB,GAAL,CAASL,OAAT,CAAZ;AACA,MAAMM,MAAMjB,KAAKiB,GAAL,CAASN,OAAT,CAAZ;AACA,MAAMO,KAAMF,OAAOF,IAAIF,EAAX,CAAD,GAAoBK,OAAOF,IAAIF,EAAX,CAApB,GAAsCD,EAAjD;AACA,MAAMO,KAAMH,OAAOD,IAAIF,EAAX,CAAD,GAAoBI,OAAOH,IAAIF,EAAX,CAApB,GAAsCC,EAAjD;;AAEA,SAAO,CAACK,EAAD,EAAKC,EAAL,CAAP;AACD;;AAED,SAASC,WAAT,CAAqBC,GAArB,EAA0BC,GAA1B,EAA+B;AAC7B,MAAMC,KAAKF,IAAI,CAAJ,IAASC,IAAI,CAAJ,CAApB;AACA,MAAME,KAAKH,IAAI,CAAJ,IAASC,IAAI,CAAJ,CAApB;AACA,MAAMG,MAAMzB,KAAK0B,IAAL,CAAUH,KAAKA,EAAL,GAAUC,KAAKA,EAAzB,CAAZ;AACA,SAAOC,GAAP;AACD;;AAEM,SAAS3D,SAAT,CAAmBuD,GAAnB,EAAwBM,GAAxB,EAA6B;AAClC,MAAMF,MAAML,YAAYC,GAAZ,EAAiBM,GAAjB,CAAZ;AACA,SAAOF,MAAM,CAAb;AACD;;AAEM,SAAS1D,QAAT,CAAkBsD,GAAlB,EAAuBM,GAAvB,EAA4B;AACjC,MAAMJ,KAAKF,IAAI,CAAJ,IAASM,IAAI,CAAJ,CAApB;AACA,MAAMH,KAAKH,IAAI,CAAJ,IAASM,IAAI,CAAJ,CAApB;AACA,MAAMF,MAAMzB,KAAK0B,IAAL,CAAUH,KAAKA,EAAL,GAAUC,KAAKA,EAAzB,CAAZ;;AAEA;AACA,MAAMrC,QAAQa,KAAK4B,IAAL,CAAUL,KAAKE,GAAf,IAAsBzB,KAAK6B,IAAL,CAAUL,EAAV,CAApC;AACA,SAAOrC,KAAP;AACD;;AAED,SAASY,aAAT,CAAuBb,MAAvB,EAA+BC,KAA/B,EAAsC;AACpC,SAAO,CAACD,SAAUc,KAAKgB,GAAL,CAAS7B,KAAT,CAAX,EAA4BD,SAAUc,KAAKiB,GAAL,CAAS9B,KAAT,CAAtC,CAAP;AACD;;AAED,SAASI,kBAAT,CAA4BuC,CAA5B,EAA+B;AAC7B,MAAMC,QAAQ/B,KAAKC,EAAL,GAAU,CAAxB;AACA,MAAM+B,MAAM,EAAZ;AACA,OAAK,IAAIpC,IAAI,CAAb,EAAgBA,IAAI,CAApB,EAAuBA,GAAvB,EAA4B;AAC1BoC,QAAI9B,IAAJ,CAASH,cAAc+B,CAAd,EAAiBC,QAAQnC,CAAzB,CAAT;AACD;;AAED,SAAOoC,GAAP;AACD;;AAED,SAASlD,cAAT,CAAwBmD,KAAxB,EAA+B;AAC7B;AACA,MAAMC,MAAM,CAAC,CAAD,EAAI,CAAJ,EAAO,CAAP,EAAU,CAAV,EAAa,CAAb,EAAgB,CAAhB,CAAZ;AACA,SAAOA,IAAInD,GAAJ,CAAQ;AAAA,WAAKkD,MAAME,CAAN,CAAL;AAAA,GAAR,CAAP;AACD;;AAED,SAAS3C,cAAT,CAAwB4C,GAAxB,EAA6BC,KAA7B,EAAoC;AAClC;AACA,MAAMC,KAAK,CAAC,CAAD,EAAI,CAAJ,CAAX;AACA,MAAM5C,cAAc,EAApB;;AAEA,OAAK,IAAIE,IAAI,CAAb,EAAgBA,IAAI,CAApB,EAAuBA,GAAvB,EAA4B;AAC1B,QAAMX,KAAKmD,IAAIxC,CAAJ,CAAX;AACA,QAAM2C,MAAMF,MAAMzC,CAAN,CAAZ;;AAEA,QAAMkC,IAAIhE,UAAUyE,GAAV,EAAeD,EAAf,CAAV;AACA,QAAMzC,KAAK/B,UAAUmB,EAAV,EAAcqD,EAAd,IAAoBR,CAA/B;;AAEA,QAAMhC,KAAKE,KAAKwC,KAAL,CAAWvD,GAAG,CAAH,CAAX,EAAkBA,GAAG,CAAH,CAAlB,IAA2Be,KAAKwC,KAAL,CAAWD,IAAI,CAAJ,CAAX,EAAmBA,IAAI,CAAJ,CAAnB,CAAtC;;AAEA7C,gBAAYQ,IAAZ,CAAiB,EAACL,MAAD,EAAKC,MAAL,EAAjB;AACD;;AAED,SAAOJ,WAAP;AACD","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"]}