2019-01-07 15:35:14 +03:00
|
|
|
import React from 'react';
|
2019-01-09 22:30:59 +03:00
|
|
|
import { Modal, ModalBody } from 'reactstrap';
|
2019-01-07 15:35:14 +03:00
|
|
|
import { Map, TileLayer, Marker, Popup } from 'react-leaflet';
|
2019-03-04 22:24:28 +03:00
|
|
|
import { prop } from 'ramda';
|
2019-01-07 15:35:14 +03:00
|
|
|
import * as PropTypes from 'prop-types';
|
|
|
|
import './MapModal.scss';
|
|
|
|
|
|
|
|
const propTypes = {
|
|
|
|
toggle: PropTypes.func,
|
|
|
|
isOpen: PropTypes.bool,
|
|
|
|
title: PropTypes.string,
|
2019-01-07 23:00:28 +03:00
|
|
|
locations: PropTypes.arrayOf(PropTypes.shape({
|
2019-01-07 23:11:09 +03:00
|
|
|
cityName: PropTypes.string.isRequired,
|
2019-01-07 23:00:28 +03:00
|
|
|
latLong: PropTypes.arrayOf(PropTypes.number).isRequired,
|
|
|
|
count: PropTypes.number.isRequired,
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
const defaultProps = {
|
|
|
|
locations: [],
|
2019-01-07 15:35:14 +03:00
|
|
|
};
|
|
|
|
|
2019-01-07 23:00:28 +03:00
|
|
|
const OpenStreetMapTile = () => (
|
|
|
|
<TileLayer
|
|
|
|
attribution='&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors'
|
|
|
|
url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
|
|
|
|
/>
|
|
|
|
);
|
2019-01-07 15:35:14 +03:00
|
|
|
|
2019-03-04 22:24:28 +03:00
|
|
|
const calculateMapProps = (locations) => {
|
2019-03-10 14:09:54 +03:00
|
|
|
if (locations.length === 0) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2019-03-04 22:24:28 +03:00
|
|
|
if (locations.length > 1) {
|
|
|
|
return { bounds: locations.map(prop('latLong')) };
|
|
|
|
}
|
|
|
|
|
|
|
|
// When there's only one location, an error is thrown if trying to calculate the bounds.
|
|
|
|
// When that happens, we use zoom and center as a workaround
|
|
|
|
const [{ latLong: center }] = locations;
|
|
|
|
|
|
|
|
return { zoom: '10', center };
|
|
|
|
};
|
2019-03-03 13:47:19 +03:00
|
|
|
|
2019-01-07 23:00:28 +03:00
|
|
|
const MapModal = ({ toggle, isOpen, title, locations }) => (
|
|
|
|
<Modal toggle={toggle} isOpen={isOpen} className="map-modal__modal" contentClassName="map-modal__modal-content">
|
|
|
|
<ModalBody className="map-modal__modal-body">
|
2019-01-09 22:30:59 +03:00
|
|
|
<h3 className="map-modal__modal-title">
|
|
|
|
{title}
|
|
|
|
<button type="button" className="close" onClick={toggle}>×</button>
|
|
|
|
</h3>
|
2019-03-04 22:24:28 +03:00
|
|
|
<Map {...calculateMapProps(locations)}>
|
2019-01-07 23:00:28 +03:00
|
|
|
<OpenStreetMapTile />
|
2019-01-07 23:11:09 +03:00
|
|
|
{locations.map(({ cityName, latLong, count }, index) => (
|
2019-01-07 23:00:28 +03:00
|
|
|
<Marker key={index} position={latLong}>
|
2019-01-07 23:11:09 +03:00
|
|
|
<Popup><b>{count}</b> visit{count > 1 ? 's' : ''} from <b>{cityName}</b></Popup>
|
2019-01-07 21:43:25 +03:00
|
|
|
</Marker>
|
2019-01-07 23:00:28 +03:00
|
|
|
))}
|
|
|
|
</Map>
|
|
|
|
</ModalBody>
|
|
|
|
</Modal>
|
|
|
|
);
|
2019-01-07 15:35:14 +03:00
|
|
|
|
|
|
|
MapModal.propTypes = propTypes;
|
2019-01-07 23:00:28 +03:00
|
|
|
MapModal.defaultProps = defaultProps;
|
2019-01-07 15:35:14 +03:00
|
|
|
|
|
|
|
export default MapModal;
|