MapView исчезает, когда вызывается animateToRegion(), а маркеры местоположения не отображаются до тех пор, пока не будет вызвана animateToRegion().
У меня проблема с загрузкой карты в мое приложение, чтобы ее маркеры загружались при первом рендере. Предполагается, что маркеры соответствуют строительным картам в виде прокрутки. Файл App.js, который вызывает CustomMapView, содержит жестко закодированные фиктивные данные в виде массива из 5 объектов Building {title, description, lat, long}, которые он передает CustomMapView, а затем CustomMapView передает данные остальным своим дочерним элементам (в код ниже)
Когда приложение загружается впервые, оно выглядит так:
Затем при прокрутке до следующего индекса появляются маркеры (и анимируются), но затем карта исчезает:
Это мой первый проект React Native, и я уже несколько дней пытаюсь понять, что маркер не рендерится и проблема исчезновения карты. Я в основном пытаюсь сделать карту с видом прокрутки, которая содержит здания, и когда вы прокручиваете эту "Карту здания", карта оживляет здание. Любая помощь будет высоко ценится. Спасибо!
class CustomMapView extends Component {
/*
this.lastLoadedIndex
this.animation
this.state
this.mapView
this.interpolations
this.regionTimeout
*/
constructor(props) {
super(props);
this.lastLoadedIndex = 0;
this.animation = new Animated.Value(0);
this.state = {
buildings: this.props.buildings,
region: this.props.initialRegion,
};
}
componentDidMount() {
this.beginAnimatingMapToMatchSelectedBuildingCard();
}
render() {
this.interpolations = this.state.buildings.map((building, index) => {
const inputRange = [(index - 1) * CARD_WIDTH, index * CARD_WIDTH, ((index + 1) * CARD_WIDTH)];
const scale = this.animation.interpolate({
inputRange,
outputRange: [INACTIVE_MARKER_RADIUS_SCALE, ACTIVE__MARKER_RADIUS_SCALE, INACTIVE_MARKER_RADIUS_SCALE],
extrapolate: "clamp",
});
const opacity = this.animation.interpolate({
inputRange,
outputRange: [INACTIVE_MARKER_OPACITY, ACTIVE_MARKER_OPACITY, INACTIVE_MARKER_OPACITY],
extrapolate: "clamp",
});
return { scale, opacity };
});
return (
<View style={StJamesMapViewStyle.container}>
<MapWithBuildingMarkers root={this} interpolations={this.interpolations} region={this.state.region} buildings={this.state.buildings} />
<BuildingCardSelectorView root={this} buildings={this.state.buildings}/>
</View>
);
}
beginAnimatingMapToMatchSelectedBuildingCard() {
this.animation.addListener(({value}) => {
let index = Math.floor(value / CARD_WIDTH + SCROLL_OFFSET);
if (index < 0)
index = 0;
else if (index >= this.state.buildings.length)
index = this.state.buildings.length - 1;
clearTimeout(this.regionTimeout);
this.regionTimeout = setTimeout(() => {
if (this.lastLoadedIndex === index)
return;
this.lastLoadedIndex = index;
const coordinateToBeLoaded = this.state.buildings[index].coordinate;
const region = {
latitude: coordinateToBeLoaded.latitude,
longitude: coordinateToBeLoaded.longitude,
latitudeDelta: this.state.region.latitudeDelta,
longitudeDelta: this.state.region.longitudeDelta
};
this.mapView.animateToRegion(region, DURATION_AFTER_BUILDING_CARD_IS_SELECTED_TO_START_ANIMATING);
console.log(region);
}, DURATION_TO_ANIMATE_TO_REGION);
});
}
}
class MapWithBuildingMarkers extends Component {
render() {
const root = this.props.root;
const buildings = this.props.buildings;
const interpolations = this.props.interpolations;
return (
<MapView ref={mapView => root.mapView = mapView} //When we animateToRegion we render MapView which has Building Markers as its children
initialRegion={this.props.region}
style={StJamesMapViewStyle.container}>
{buildings.map((building, index) => <BuildingMarker key={index} index={index} building={building} interpolations={interpolations}/>)}
</MapView>);
}
}
class BuildingMarker extends Component {
render() {
const scaleStyle = {transform: [{scale: this.props.interpolations[this.props.index].scale}]};
const opacityStyle = {opacity: this.props.interpolations[this.props.index].opacity};
return (
<MapView.Marker coordinate={this.props.building.coordinate}>
<AnimatedView style={[StJamesMapViewStyle.markerWrap, opacityStyle]}>
<AnimatedView style={[StJamesMapViewStyle.ring, scaleStyle]}/>
<View style={StJamesMapViewStyle.marker} />
</AnimatedView>
</MapView.Marker>);
}
}
class BuildingCardSelectorView extends Component {
render() {
const root = this.props.root;
const scrollHandler = Animated.event([{
nativeEvent: {
contentOffset: {
x: root.animation,
}
}}],
{ useNativeDriver: true }
);
return (
<Animated.ScrollView horizontal
scrollEventThrottle={1}
showsHorizontalScrollIndicator={false}
snapToInterval={CARD_WIDTH}
onScroll={scrollHandler}
style={StJamesMapViewStyle.scrollView}
contentContainerStyle={StJamesMapViewStyle.endPadding}>
{this.props.buildings.map((building, index) => <BuildingCard key={index}
index={index}
building={building}/>)}
</Animated.ScrollView>
);
}
}
class BuildingCard extends Component {
render() {
return (
<View style={StJamesMapViewStyle.card}>
<Image source={this.props.building.imageURL} style={StJamesMapViewStyle.cardImage} resizeMode="cover"/>
<View style={StJamesMapViewStyle.textContent}>
<Text numberOfLines={1}> {this.props.building.title} </Text>
<Text numberOfLines={1}> {this.props.building.description} </Text>
</View>
</View>
);
}
}