问题
I'm working with the Citybik.es API (http://api.citybik.es/) to show data on a Leaflet map.
At the moment, the code is showing every item in the map, but after analysing I would like to organize the data a little bit allowing the user to drill down the information, when clicking on a marker. Something like this:
- Display number of networks per country (each marker represents a network, so to speak, each marker per country)
- Display number of stations per network
The response looks something like this:
Here's the JavaScript:
import React, { Component } from 'react';
import L from 'leaflet';
import { Map, TileLayer, Marker, Popup } from 'react-leaflet';
// code for map marker icon
var myIcon = L.icon({
iconUrl: '',
iconSize: [25, 41],
iconAnchor: [12.5, 41],
popupAnchor: [0, -41]
});
class App extends Component {
state = {
location: {
lat: 51.505,
lng: -0.09,
},
bikeData: [],
haveUsersLocation: false,
zoom: 3,
}
//lifecycle method to get the bike information
componentDidMount() {
fetch('https://api.citybik.es/v2/networks')
.catch(error => {
console.log(error)
})
.then(res => res.json())
.then(response => {
const networkData = response.networks;
this.setState({
bikeData: networkData
});
})
}
render() {
const position = [this.state.location.lat, this.state.location.lng]
const bikeData = this.state.bikeData;
return (
<Map className="map" center={position} zoom={this.state.zoom}>
<TileLayer
attribution="&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors"
url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
/>
{
bikeData && bikeData.map((data) => {
console.log(data)
return (
<Marker
icon={myIcon}
key={data.id}
position={[data.location.latitude, data.location.longitude]}>
<Popup>
Name: {data.name} <br />
Station Details: {[data.location.city, data.location.country]}
</Popup>
</Marker>
)
})
}
</Map>
)
}
}
ReactDOM.render(<App/>,
document.getElementById('root')
);
Thank you!
回答1:
Maybe something like this to get the count
const networkCountByCountry = new Map();
response.networks.forEach(network => {
const country = network.location.country
const countryObject = networkCountByCountry.has(country) ? networkCountByCountry.get(country) : {
id: country,
count: 0
};
countryObject.count = countryObject.count + 1
networkCountByCountry.set(countryObject.id, countryObject)
})
const networkCountByCountryArray = [...networkCountByCountry.values()]
Then, you can use a click event to filter to the selected country list.
A more complete example:
import L from "leaflet";
import { Map as MapLeaf, TileLayer, Marker, Popup } from "react-leaflet";
// code for map marker icon
var myIcon = L.icon({
iconUrl:
"",
iconSize: [25, 41],
iconAnchor: [12.5, 41],
popupAnchor: [0, -41]
});
class MapDisplay extends Component {
state = {
location: {
lat: 51.505,
lng: -0.09
},
selectedCountry: null,
bikeData: [],
haveUsersLocation: false,
zoom: 3
};
//lifecycle method to get the bike information
componentDidMount() {
fetch("https://api.citybik.es/v2/networks")
.catch(error => {
console.log(error);
})
.then(res => res.json())
.then(response => {
const networkData = response.networks;
const networkCountByCountry = new Map();
response.networks.forEach(network => {
const country = network.location.country;
const countryObject = networkCountByCountry.has(country)
? networkCountByCountry.get(country)
: { id: country, count: 0, location: network.location };
countryObject.count = countryObject.count + 1;
networkCountByCountry.set(countryObject.id, countryObject);
});
const networkCountByCountryArray = [...networkCountByCountry.values()];
this.setState({
bikeData: networkData,
bikeCountryData: networkCountByCountryArray
});
});
}
countryMarkerClick(country) {
console.log(country);
this.setState({ selectedCountry: country });
}
render() {
const position = [this.state.location.lat, this.state.location.lng];
if (this.state.selectedCountry) {
const bikeData = this.state.bikeData.filter(
network => network.location.country === this.state.selectedCountry
);
return (
<MapLeaf className="map" center={position} zoom={this.state.zoom}>
<TileLayer
attribution="&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors"
url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
/>
{bikeData &&
bikeData.map(data => {
console.log(data);
return (
<Marker
icon={myIcon}
key={data.id}
position={[data.location.latitude, data.location.longitude]}
>
<Popup>
Name: {data.name} <br />
Station Details:{" "}
{[data.location.city, data.location.country]}
</Popup>
</Marker>
);
})}
</MapLeaf>
);
}
const bikeCountryData = this.state.bikeCountryData;
return (
<MapLeaf className="map" center={position} zoom={this.state.zoom}>
<TileLayer
attribution="&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors"
url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
/>
{bikeCountryData &&
bikeCountryData.map(data => {
return (
<Marker
icon={myIcon}
key={data.id}
position={[data.location.latitude, data.location.longitude]}
>
<Popup>
Name: {data.id} <br />
Stations: {[data.count]} <br />
<span onClick={this.countryMarkerClick.bind(this, data.id)}>
{" "}
-->{" "}
</span>
</Popup>
</Marker>
);
})}
</MapLeaf>
);
}
}
来源:https://stackoverflow.com/questions/52557802/how-to-get-the-number-of-map-markers-per-country-reactjs