mirror of
https://github.com/plausible/analytics.git
synced 2025-01-03 15:17:58 +03:00
22a2eae940
The more links are absolutely positioned. It works fine on desktop when there is two columns. On mobile, the absolute sends the more links on top of each other on top of the column. Adding a simple "relative" class on each cards that contains a more link ensures that they stay when they belong.
106 lines
2.9 KiB
JavaScript
106 lines
2.9 KiB
JavaScript
import React from 'react';
|
|
import Datamap from 'datamaps'
|
|
|
|
import FadeIn from '../fade-in'
|
|
import Bar from './bar'
|
|
import MoreLink from './more-link'
|
|
import * as api from '../api'
|
|
|
|
export default class Countries extends React.Component {
|
|
constructor(props) {
|
|
super(props)
|
|
this.resizeMap = this.resizeMap.bind(this)
|
|
this.state = {
|
|
loading: true
|
|
}
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.fetchCountries()
|
|
window.addEventListener('resize', this.resizeMap);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
window.removeEventListener('resize', this.resizeMap);
|
|
}
|
|
|
|
componentDidUpdate(prevProps) {
|
|
if (this.props.query !== prevProps.query) {
|
|
this.setState({loading: true, countries: null})
|
|
this.fetchCountries()
|
|
}
|
|
}
|
|
|
|
fetchCountries() {
|
|
api.get(`/api/stats/${encodeURIComponent(this.props.site.domain)}/countries`, this.props.query)
|
|
.then((res) => this.setState({loading: false, countries: res}))
|
|
.then(() => this.drawMap())
|
|
}
|
|
|
|
resizeMap() {
|
|
this.map && this.map.resize()
|
|
}
|
|
|
|
drawMap() {
|
|
var dataset = {};
|
|
|
|
var onlyValues = this.state.countries.map(function(obj){ return obj.count });
|
|
var minValue = Math.min.apply(null, onlyValues),
|
|
maxValue = Math.max.apply(null, onlyValues);
|
|
|
|
var paletteScale = d3.scale.linear()
|
|
.domain([minValue,maxValue])
|
|
.range(["#f3ebff","#a779e9"]);
|
|
|
|
this.state.countries.forEach(function(item){
|
|
dataset[item.name] = {numberOfThings: item.count, fillColor: paletteScale(item.count)};
|
|
});
|
|
|
|
this.map = new Datamap({
|
|
element: document.getElementById('map-container'),
|
|
responsive: true,
|
|
projection: 'mercator',
|
|
fills: { defaultFill: '#f8fafc' },
|
|
data: dataset,
|
|
geographyConfig: {
|
|
borderColor: '#dae1e7',
|
|
highlightBorderWidth: 2,
|
|
highlightFillColor: function(geo) {
|
|
return geo['fillColor'] || '#F5F5F5';
|
|
},
|
|
highlightBorderColor: '#a779e9',
|
|
popupTemplate: function(geo, data) {
|
|
if (!data) { return ; }
|
|
return ['<div class="hoverinfo">',
|
|
'<strong>', geo.properties.name, '</strong>',
|
|
'<br><strong>', data.numberOfThings, '</strong> Visitors',
|
|
'</div>'].join('');
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
renderBody() {
|
|
if (this.state.countries) {
|
|
return (
|
|
<React.Fragment>
|
|
<h3 className="font-bold">Countries</h3>
|
|
<div className="mt-6 mx-auto" style={{width: '100%', maxWidth: '475px', height: '320px'}} id="map-container"></div>
|
|
<MoreLink site={this.props.site} list={this.state.countries} endpoint="countries" />
|
|
</React.Fragment>
|
|
)
|
|
}
|
|
}
|
|
|
|
render() {
|
|
return (
|
|
<div className="stats-item relative bg-white shadow-xl rounded p-4" style={{height: '436px'}}>
|
|
{ this.state.loading && <div className="loading my-32 mx-auto"><div></div></div> }
|
|
<FadeIn show={!this.state.loading}>
|
|
{ this.renderBody() }
|
|
</FadeIn>
|
|
</div>
|
|
)
|
|
}
|
|
}
|