How to get the number of map markers per country - ReactJS

China☆狼群 提交于 2019-12-12 10:22:48

问题


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:

  1. Display number of networks per country (each marker represents a network, so to speak, each marker per country)
  2. 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="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>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="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>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="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>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

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!