Я пытаюсь заставить мое приложение реагировать на родные карты взаимодействовать с API-интерфейсом геокодирования arcgis esri.По какой-то причине, когда я меняю регион, прокрутка вокруг маркеров очень противоречива и будет случайным образом всплывать и исчезать, и в целом будет очень глючной.Есть ли что-то, что я должен сделать, чтобы перерисовать только определенные метки или как-то кэшировать маркеры в текущем регионе?
Демо : https://giphy.com/gifs/1o1o3wktAFkcV7B06R
import React from 'react';
import { Dimensions, ScrollView, StyleSheet, View } from 'react-native';
import { ExpoLinksView } from '@expo/samples';
import MapView, { Marker } from 'react-native-maps';
import SphericalMercator from "@mapbox/sphericalmercator";
const dimensions = Dimensions.get('window');
const ASPECT_RATIO = dimensions.width / dimensions.height;
const LATITUDE = 41.8241;
const LONGITUDE = -87.6561;
const LATITUDE_DELTA = 0.003;
const LONGITUDE_DELTA = LATITUDE_DELTA * ASPECT_RATIO;
const merc = new SphericalMercator();
const getZoomLevelFromRegion = (region, viewport) => {
const { longitudeDelta } = region;
const { width } = viewport;
const lngD = (360 + longitudeDelta) % 360;
const tiles = width / 256;
const portion = lngD / 360;
const tilePortion = portion / tiles;
return Math.log2(1 / tilePortion);
};
const getRegionByZoomLevel = (center, zoomLevel, viewport) => {
const { latitude, longitude } = center;
const { width, height } = viewport;
const [x, y] = merc.px([lng, lat], zoomLevel);
const xmin = Math.floor(x - width / 2);
const xmax = xmin + width;
const ymin = Math.floor(y - height / 2);
const ymax = ymin + height;
const nw = merc.ll([xmin, ymin], zoomLevel);
const se = merc.ll([xmax, ymax], zoomLevel);
const latitudeDelta = nw[1] - se[1];
const longitudeDelta = se[0] - nw[0];
return {
latitude,
longitude,
latitudeDelta,
longitudeDelta
};
};
const getRegionBoundaries = (center, zoomLevel, viewport ) => {
const { latitude, longitude, latitudeDelta, longitudeDelta } = center;
const { width, height } = viewport;
const [x, y] = merc.px([longitude, latitude], zoomLevel);
const xmin = Math.floor(x - width / 2);
const xmax = xmin + width;
const ymin = Math.floor(y - height / 2);
const ymax = ymin + height;
const [westLongitude, northLatitude] = merc.ll([xmin, ymin], zoomLevel);
const [eastLongitude, southLatitude] = merc.ll([xmax, ymax], zoomLevel);
return {
northLatitude,
southLatitude,
westLongitude,
eastLongitude
};
};
export default class LinksScreen extends React.Component {
constructor(props) {
super(props);
this.state = {
region: {
latitude: LATITUDE,
longitude: LONGITUDE,
latitudeDelta: LATITUDE_DELTA,
longitudeDelta: LONGITUDE_DELTA,
},
layout: null,
markers: [],
};
this.onRegionChange = this.onRegionChange.bind(this);
this.getZoomLevel = this.getZoomLevel.bind(this);
}
getZoomLevel() {
const { region, layout } = this.state;
if (!region || !layout) return null;
return getZoomLevelFromRegion(region,layout);
}
onRegionChange(region) {
const zoomLevel = this.getZoomLevel();
if (!zoomLevel) {
return null;
}
const { latitude, longitude } = region;
const { width, height } = this.state.layout;
const {
northLatitude,
southLatitude,
westLongitude,
eastLongitude
} = getRegionBoundaries(
{ latitude, longitude },
Math.floor(zoomLevel) + 1,
{ width: width, height: height }
);
if(zoomLevel < 13){
if(this.state.markers.length){
this.setState({
region,
layout: this.state.layout,
markers: [],
});
}
return;
}
fetch(`https://some_arcgis_api/query?geometry={"xmin":${westLongitude},"ymin":${southLatitude},"xmax":${eastLongitude}",ymax":${northLatitude},"spatialReference":{"wkid":4326}}&geometryType=esriGeometryEnvelope&spatialRel=esriSpatialRelIntersects&geometryPrecision=6&f=geojson`)
.then((response) => response.json())
.then((responseJSON) => {
this.setState({
region,
layout: this.state.layout,
markers: responseJSON.features.map(function(feature){
return {
key: feature.id,
coordinate: {
latitude: feature.geometry.coordinates[1],
longitude: feature.geometry.coordinates[0],
}
}
})
});
})
}
static navigationOptions = {
header: null,
};
render() {
return (
<View style={styles.container}>
<MapView
style={styles.map}
mapType="hybrid"
onLayout={e => this.setState({
layout: {width: e.nativeEvent.layout.width * 2,height:e.nativeEvent.layout.height * 2},
region: this.state.region,
markers: this.state.markers,
})}
initialRegion={{
latitude: LATITUDE,
longitude: LONGITUDE,
latitudeDelta: LATITUDE_DELTA,
longitudeDelta: LONGITUDE_DELTA,
}}
onRegionChange={this.onRegionChange}
>
{this.state.markers.map(marker => (
<Marker
key={marker.key}
coordinate={marker.coordinate}
/>
))}
</MapView>
</View>
);
}
}
const styles = StyleSheet.create({
container: {
...StyleSheet.absoluteFillObject,
justifyContent: 'flex-end',
alignItems: 'center',
},
map: {
...StyleSheet.absoluteFillObject,
},
});