UNPKG

sg-heatmap

Version:

Open-source all-in-one Swiss Army knife tool for creating Choropleth maps

358 lines (306 loc) 13.2 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.Feature = undefined; var _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; }; var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _minBy = require('lodash/minBy'); var _minBy2 = _interopRequireDefault(_minBy); var _maxBy = require('lodash/maxBy'); var _maxBy2 = _interopRequireDefault(_maxBy); var _cloneDeep = require('lodash/cloneDeep'); var _cloneDeep2 = _interopRequireDefault(_cloneDeep); var _isEqual = require('lodash/isEqual'); var _isEqual2 = _interopRequireDefault(_isEqual); var _partition3 = require('lodash/partition'); var _partition4 = _interopRequireDefault(_partition3); var _geometry2 = require('./helpers/geometry'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var SgHeatmap = function () { function SgHeatmap(data) { _classCallCheck(this, SgHeatmap); var _data = typeof data === 'string' ? JSON.parse(data) : data; if (!(_data instanceof Array)) throw new Error('Expects an array of feature objects'); this.children = _data.map(function (f) { return new Feature(f); }); this._defaultState = {}; this._updaters = []; this._stats = {}; } _createClass(SgHeatmap, [{ key: 'setDefaultState', value: function setDefaultState(_key, value) { this._defaultState[_key] = value; this.children.forEach(function (c) { if (_key in c.state) return; c.state[_key] = (0, _cloneDeep2.default)(value); }); return this; } }, { key: 'resetState', value: function resetState() { var _this = this; this.children.forEach(function (c) { c.state = (0, _cloneDeep2.default)(_this._defaultState); }); return this; } }, { key: 'registerUpdater', value: function registerUpdater(fn) { this._updaters.push(fn); return this; } }, { key: 'inspectUpdaters', value: function inspectUpdaters() { this._updaters.map(function (fn, i) { console.log('Inspecting updater "' + (i + 1) + '"'); console.log(fn); }); } }, { key: 'registerStat', value: function registerStat(key, fn) { this._stats[key] = fn; return this; } }, { key: 'inspectStats', value: function inspectStats() { var _this2 = this; Object.keys(this._stats).forEach(function (key, i) { console.log('Inspecting stat "' + key + '"'); console.log(_this2._stats[key]); }); } }, { key: 'bin', value: function bin(lnglat) { return this.children.filter(function (c) { return c.inside(lnglat); }); } }, { key: 'update', value: function update(lnglat, weight) { var _this3 = this; this.bin(lnglat).forEach(function (c) { c.state = _this3._updaters.reduce(function (nextState, fn) { return Object.assign(nextState, fn(weight, c.state)); }, {}); }); return this; } }, { key: 'getStat', value: function getStat(stat) { var _this4 = this; var fn = typeof stat === 'function' ? stat : this._stats[stat]; var _partition = (0, _partition4.default)(this.children, function (c) { return !(0, _isEqual2.default)(_this4._defaultState, c.state); }), _partition2 = _slicedToArray(_partition, 2), changed = _partition2[0], unchanged = _partition2[1]; var listedValues = []; var values = changed.reduce(function (stats, c) { var value = fn(c.state, c.properties); listedValues.push(value); return Object.assign(stats, _defineProperty({}, c.id, value)); }, {}); return { stat: stat, values: values, unchanged: unchanged.map(function (c) { return c.id; }), min: (0, _minBy2.default)(listedValues), max: (0, _maxBy2.default)(listedValues) }; } }, { key: 'initializeRenderer', value: function initializeRenderer(colorScale) { var _this5 = this; var defaultStyle = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var addonStyle = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; if (!window) throw new Error('Method initializeRenderer should only be called browser-side'); if (!window.google) throw new Error('Google Maps not loaded'); if ('renderer' in this) { console.log('Existing renderer replaced'); this.renderer.setMap(null); } this.colorScale = colorScale; this.renderer = new window.google.maps.Data({ style: function style(feature) { var styleOptions = Object.assign({}, defaultStyle); var color = feature.getProperty('color'); if (color) Object.assign(styleOptions, addonStyle, { fillColor: color }); return styleOptions; } }); this.children.forEach(function (c) { _this5.renderer.addGeoJson({ id: c.id, type: 'Feature', geometry: c.geometry, properties: Object.assign({}, c.properties, { color: null }) }); }); return this.renderer; } }, { key: 'render', value: function render(stat) { var _this6 = this; var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; if (!this.renderer) throw new Error('Renderer has not been initialized'); var colorScale = options.colorScale || this.colorScale; var _getStat = this.getStat(stat), statValues = _getStat.values, unchanged = _getStat.unchanged, min = _getStat.min, max = _getStat.max; var domain = options.domain || [min, max]; function normalize(value) { return (value - domain[0]) / (domain[1] - domain[0]); } Object.keys(statValues).forEach(function (key) { var normalized = normalize(statValues[key]); var transformed = Math.pow(normalized, options.transform || 1); var color = colorScale(transformed); _this6.renderer.getFeatureById(key).setProperty('color', color); }); unchanged.forEach(function (key) { _this6.renderer.getFeatureById(key).setProperty('color', null); }); } }, { key: 'clone', value: function clone() { var includeState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false; var cloned = new SgHeatmap(this.children); cloned._updaters = [].concat(_toConsumableArray(this._updaters)); cloned._stats = _extends({}, this._stats); if (includeState) { cloned._defaultState = this._defaultState; } else { cloned.children.forEach(function (c) { c.state = {}; }); } return cloned; } }, { key: 'serialize', value: function serialize() { var includeState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false; return '[' + this.children.map(function (c) { return c.serialize(includeState); }).join(',') + ']'; } }]); return SgHeatmap; }(); exports.default = SgHeatmap; var Feature = exports.Feature = function () { function Feature(data) { _classCallCheck(this, Feature); if (!('id' in data)) throw new Error('Feature object requires id'); if (!('geometry' in data)) throw new Error('Geometry not specified in feature object'); this.type = 'Feature'; this.id = data.id; this.properties = data.properties ? (0, _cloneDeep2.default)(data.properties) : {}; this.geometry = {}; this.geometry.type = data.geometry.type; if (this.geometry.type === 'Polygon') { this.geometry.coordinates = data.geometry.coordinates.map(_geometry2.toLinearRing); this.geometry.bbox = 'bbox' in data.geometry ? (0, _cloneDeep2.default)(data.geometry.bbox) : [(0, _minBy2.default)(this.geometry.coordinates[0], function (v) { return v[0]; })[0], (0, _minBy2.default)(this.geometry.coordinates[0], function (v) { return v[1]; })[1], (0, _maxBy2.default)(this.geometry.coordinates[0], function (v) { return v[0]; })[0], (0, _maxBy2.default)(this.geometry.coordinates[0], function (v) { return v[1]; })[1]]; } else if (this.geometry.type === 'MultiPolygon') { this.geometry.coordinates = data.geometry.coordinates.map(function (polygon) { return polygon.map(_geometry2.toLinearRing); }); if ('bbox' in data.geometry) { this.geometry.bbox = (0, _cloneDeep2.default)(data.geometry.bbox); } else { var bboxs = this.geometry.coordinates.map(function (polygon) { return [(0, _minBy2.default)(polygon[0], function (v) { return v[0]; })[0], (0, _minBy2.default)(polygon[0], function (v) { return v[1]; })[1], (0, _maxBy2.default)(polygon[0], function (v) { return v[0]; })[0], (0, _maxBy2.default)(polygon[0], function (v) { return v[1]; })[1]]; }); this.geometry.bbox = [(0, _minBy2.default)(bboxs, function (v) { return v[0]; })[0], (0, _minBy2.default)(bboxs, function (v) { return v[1]; })[1], (0, _maxBy2.default)(bboxs, function (v) { return v[2]; })[2], (0, _maxBy2.default)(bboxs, function (v) { return v[3]; })[3]]; } } else { throw new Error('Feature geometry must be of type Polygon or MultiPolygon'); } this.state = 'state' in data ? (0, _cloneDeep2.default)(data.state) : {}; } _createClass(Feature, [{ key: 'inside', value: function inside(location) { var _location = _slicedToArray(location, 2), lng = _location[0], lat = _location[1]; if (lng < this.geometry.bbox[0]) return false; if (lat < this.geometry.bbox[1]) return false; if (lng > this.geometry.bbox[2]) return false; if (lat > this.geometry.bbox[3]) return false; if (this.geometry.type === 'Polygon') { return (0, _geometry2.inside)([lng, lat], this.geometry.coordinates[0]); } else { return this.geometry.coordinates.some(function (polygon) { return (0, _geometry2.inside)([lng, lat], polygon[0]); }); } } }, { key: 'serialize', value: function serialize() { var includeState = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false; var id = this.id, properties = this.properties, geometry = this.geometry, state = this.state; var _geometry = { type: geometry.type, bbox: geometry.bbox, coordinates: geometry.type === 'Polygon' ? geometry.coordinates.map(_geometry2.encodePolyline) : geometry.coordinates.map(function (polygon) { return polygon.map(_geometry2.encodePolyline); }) }; var _state = includeState ? state : {}; return JSON.stringify({ id: id, properties: properties, geometry: _geometry, state: _state }); } }]); return Feature; }();