如何获取每个国家的地图标记数量-ReactJS

时间:2018-09-28 14:47:55

标签: json reactjs api leaflet react-leaflet

我正在使用Citybik.es API(http://api.citybik.es/)在传单地图上显示数据。

enter image description here

此刻,代码正在显示地图中的每个项目,但是经过分析后,我想对数据进行一些整理,以便用户在点击标记时可以深入了解信息 。像这样:

  1. 显示每个国家/地区的网络数量(每个标记代表一个网络,可以说每个国家/地区的每个标记)
  2. 显示每个网络的站数

响应看起来像这样:

enter image description here

这是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')
);

谢谢!

1 个答案:

答案 0 :(得分:0)

也许是这样的事情

  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()]

然后,您可以使用单击事件来过滤到选定的国家/地区列表。

更完整的示例:

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>
    );
  }
}