Как нарисовать линию навигации на карте mapbox в react-native?

#android #ios #react-native #react-native-mapbox-gl

#Android #iOS #react-native #react-native-mapbox-gl

Вопрос:

Я пытаюсь получить навигационные указания, используя mapbox-sdk для react-native из пакета npm:

«@mapbox / mapbox-sdk»: «^ 0.11.0»

И для рендеринга направлений, возвращаемых mapbox-sdk, я использую приведенный ниже пакет npm:

«@react-native-mapbox-gl / maps»: «^ 8.1.0-rc.8»,

Код, который я использую для получения указаний:

 import MapboxGL from '@react-native-mapbox-gl/maps'
// Mapbox SDK related package
import MapboxDirectionsFactory from '@mapbox/mapbox-sdk/services/directions'
import { lineString as makeLineString } from '@turf/helpers'
import GeoLocationService from '../../services/geolocation/GeoLocationService';
import GeoLocationCore from '@react-native-community/geolocation'

const accessToken = "ACESS_TOKEN_FROM_MAPBOX_API_DASHBOARD"
const directionsClient = MapboxDirectionsFactory({accessToken})


  constructor(props) {
    super(props);
    this.state = {
        longitude: 0,
        latitude: 0,
        orderLongitude: 0,
        orderLatitude: 0,
        route: null,
    };
  }

async componentDidMount() {
      
      const {route} = this.props

      // Lets say route.params contains the below object:
      // { "longitude": "33.981982", "latitude": "-6.851599"}
      console.log("Params from other screen: ", route.params)

      MapboxGL.setAccessToken(accessToken)
      MapboxGL.setConnected(true);
      MapboxGL.setTelemetryEnabled(true);
      const permission = await MapboxGL.requestAndroidLocationPermissions();

      let latitude, longitude;
      if(Platform.OS == "android") {
          GeoLocationService.requestLocationPermission().then(() => {
            GeoLocationCore.getCurrentPosition(
                info => {
                    const { coords } = info
    
                    latitude = coords.latitude
                    longitude = coords.longitude
                    
                    //this.setState({longitude: coords.longitude, latitude: coords.latitude})
                    this.setState({longitude: -6.873795, latitude: 33.990777, orderLongitude: route.params.longitude, orderLatitude: route.params.latitude})
                    console.log("your lon: ", longitude)
                    console.log("your lat", latitude)
                    this.getDirections([-6.873795, 33.990777], [route.params.longitude, route.params.latitude])

                },
                error => console.log(error),
                {
                    enableHighAccuracy: false,
                    //timeout: 2000,
                    maximumAge: 3600000
                }
            )
        })
      }
  }

  getDirections = async (startLoc, destLoc) => {
      const reqOptions = {
        waypoints: [
          {coordinates: startLoc},
          {coordinates: destLoc},
        ],
        profile: 'driving',
        geometries: 'geojson',
      };
      const res = await directionsClient.getDirections(reqOptions).send()
      //const route = makeLineString(res.body.routes[0].geometry.coordinates)
      const route = makeLineString(res.body.routes[0].geometry.coordinates)
      console.log("Route: ", JSON.stringify(route))
      this.setState({route: route})
      
  }
  

Код, который я использую для рендеринга дорожных указателей, выбранных mapbox-sdk:

   renderRoadDirections = () => {
    const { route } = this.state
    return route ? (
      <MapboxGL.ShapeSource id="routeSource" shape={route.geometry}>
        <MapboxGL.LineLayer id="routeFill" aboveLayerID="customerAnnotation" style={{lineColor: "#ff8109", lineWidth: 3.2, lineCap: MapboxGL.LineJoin.Round, lineOpacity: 1.84}} />
      </MapboxGL.ShapeSource>
    ) : null;
  };
  

Код, который я использую для отображения карты и направлений:

 render() {
   return (
       <View style={{ flex: 1 }}>
         <MapboxGL.MapView
                ref={(c) => this._map = c}
                style={{flex: 1, zIndex: -10}}
                styleURL={MapboxGL.StyleURL.Street}
                zoomLevel={10}
                showUserLocation={true}
                userTrackingMode={1}
                centerCoordinate={[this.state.longitude, this.state.latitude]}
                logoEnabled={true}
                >
                    {this.renderRoadDirections()}
            <MapboxGL.Camera
                zoomLevel={10}
                centerCoordinate={[this.state.longitude, this.state.latitude]}
                animationMode="flyTo"
                animationDuration={1200}
            />
        </MapboxGL.MapView>
       </View>
    )
}
  

Теперь, когда я пытаюсь отобразить GeoJSON, на карте не отображается линия направления дороги, поэтому я подумал, может быть, что-то не так с моим GeoJSON, и протестировал его отсюда, но он выглядит нормально:

https://geojsonlint.com/

GeoJSON, который я тестировал и выглядит нормально:

 {"type":"Feature","properties":{},"geometry":{"type":"LineString","coordinates":[[-6.880611,33.9916],[-6.882194,33.990166],[-6.882439,33.99015],[-6.882492,33.990028],[-6.882405,33.98991],[-6.878006,33.990299],[-6.87153,33.990978],[-6.871386,33.990925],[-6.871235,33.991016],[-6.869793,33.991165],[-6.870523,33.990292]]}}
  

Пример того, чего я пытаюсь достичь:

введите описание изображения здесь

Что может быть не так в моем коде, из-за чего линия направления дороги не отображается на карте?

Комментарии:

1. эй, что произойдет, если вместо этого вы измените его на this? <MapboxGL.ShapeSource id="routeSource" shape={route.geometry.coordinates}> ?

2. @flaky Я думаю, что он ожидает объект shape, но я пробовал, это не сработало.

Ответ №1:

Найдено, что вызывало <LineLayer/> отсутствие отображения на карте, удаление атрибута aboveLayerID из следующей строки:

 <MapboxGL.LineLayer id="routeFill" aboveLayerID="customerAnnotation" style={{lineColor: "#ff8109", lineWidth: 3.2, lineCap: MapboxGL.LineJoin.Round, lineOpacity: 1.84}} /> 
  

Таким образом, становится:

 <MapboxGL.LineLayer id="routeFill" style={{lineColor: "#ff8109", lineWidth: 3.2, lineCap: MapboxGL.LineJoin.Round, lineOpacity: 1.84}} />
  

Результат:

введите описание изображения здесь