import React from 'react'; import Datamap from 'datamaps' import { withRouter } from 'react-router-dom' import * as d3 from "d3" import numberFormatter from '../../util/number-formatter' import FadeIn from '../../fade-in' import LazyLoader from '../../components/lazy-loader' import MoreLink from '../more-link' import * as api from '../../api' import { navigateToQuery } from '../../query' class Countries extends React.Component { constructor(props) { super(props) this.resizeMap = this.resizeMap.bind(this) this.drawMap = this.drawMap.bind(this) this.getDataset = this.getDataset.bind(this) this.state = { loading: true, darkTheme: document.querySelector('html').classList.contains('dark') || false } this.onVisible = this.onVisible.bind(this) } componentDidUpdate(prevProps) { if (this.props.query !== prevProps.query) { // eslint-disable-next-line react/no-did-update-set-state this.setState({loading: true, countries: null}) this.fetchCountries().then(this.drawMap) } } componentWillUnmount() { window.removeEventListener('resize', this.resizeMap); } onVisible() { this.fetchCountries().then(this.drawMap.bind(this)) window.addEventListener('resize', this.resizeMap); if (this.props.timer) this.props.timer.onTick(this.updateCountries.bind(this)) } getDataset() { const dataset = {}; var onlyValues = this.state.countries.map(function(obj){ return obj.visitors }); var maxValue = Math.max.apply(null, onlyValues); // eslint-disable-next-line no-undef const paletteScale = d3.scale.linear() .domain([0,maxValue]) .range([ this.state.darkTheme ? "#2e3954" : "#f3ebff", this.state.darkTheme ? "#6366f1" : "#a779e9" ]) this.state.countries.forEach(function(item){ dataset[item.code] = {numberOfThings: item.visitors, fillColor: paletteScale(item.visitors)}; }); return dataset } updateCountries() { this.fetchCountries().then(() => { this.map.updateChoropleth(this.getDataset(), {reset: true}) }) } fetchCountries() { return api.get(`/api/stats/${encodeURIComponent(this.props.site.domain)}/countries`, this.props.query, {limit: 300}) .then((res) => this.setState({loading: false, countries: res})) } resizeMap() { this.map && this.map.resize() } drawMap() { const dataset = this.getDataset(); const label = this.props.query.period === 'realtime' ? 'Current visitors' : 'Visitors' const defaultFill = this.state.darkTheme ? '#2d3747' : '#f8fafc' const highlightFill = this.state.darkTheme ? '#374151' : '#F5F5F5' const borderColor = this.state.darkTheme ? '#1f2937' : '#dae1e7' const highlightBorderColor = this.state.darkTheme ? '#4f46e5' : '#a779e9' this.map = new Datamap({ element: document.getElementById('map-container'), responsive: true, projection: 'mercator', fills: { defaultFill }, data: dataset, geographyConfig: { borderColor, highlightBorderWidth: 2, highlightFillColor: (geo) => geo.fillColor || highlightFill, highlightBorderColor, popupTemplate: (geo, data) => { if (!data) { return null; } const pluralizedLabel = data.numberOfThings === 1 ? label.slice(0, -1) : label return ['
', '', geo.properties.name, ' ', '
', numberFormatter(data.numberOfThings), ' ', pluralizedLabel, '
'].join(''); } }, done: (datamap) => { datamap.svg.selectAll('.datamaps-subunit').on('click', (geography) => { navigateToQuery( this.props.history, this.props.query, { country: geography.id, country_name: geography.properties.name } ) }) } }); } geolocationDbNotice() { if (this.props.site.selfhosted) { return ( IP Geolocation by DB-IP ) } return null } renderBody() { if (this.state.countries) { return ( <>
{ this.geolocationDbNotice() } ) } return null } render() { return ( { this.state.loading &&
} { this.renderBody() }
) } } export default withRouter(Countries)