MapView не показывает карту Google в привязке данных Android mvvm [РЕШЕНО]

#android #mvvm #google-maps-android-api-2 #android-mapview #android-databinding

#Android #mvvm #google-maps-android-api-2 #android-mapview #android-привязка данных

Вопрос:

Я пытаюсь настроить карты Google с помощью MapView с привязкой данных и установить некоторые маркеры на карте. Моя карта работает нормально, когда я жестко задаю «latlngs», но когда я пытаюсь установить «latlngs», которые я получаю с сервера, карта не отображается. Отображаются маркеры, проблем с данными и настройкой маркеров нет, не работает только карта. И логотип Google есть, если вам интересно, действителен ли мой ключ API. Вот мой код.

Это фрагмент

 private MapView google_map;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

}

@Override
public View onCreateView(@NotNull LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {

    FragmentMapBinding binding = DataBindingUtil.inflate(inflater, R.layout.fragment_map, container, false);
    MapFragmentViewModel viewModel = new ViewModelProvider(this, new MapFragmentViewModelFactory()).get(MapFragmentViewModel.class);

    binding.setLifecycleOwner(this);
    binding.setViewModel(viewModel);
    
    google_map = binding.googleMap;

    viewModel.getLocations();
    viewModel.locationsResponse().observe(getViewLifecycleOwner(), apiResponseObject -> {
        switch (apiResponseObject.status) {
            case SUCCESS:
                ViewUtils.hideProgressDialog();
                break;
            case LOADING:
                ViewUtils.showProgressDialog(requireContext());
                break;
            case ERROR:
                ViewUtils.hideProgressDialog();
                ViewUtils.showToastMessage(requireContext(), apiResponseObject.error);
        }
    });

    setTypeFace(binding.getRoot(), requireContext());
    languageSetup(requireContext());
    return binding.getRoot();
}

@Override
public void onResume() {
    super.onResume();
    google_map.onResume();
}

@Override
public void onPause() {
    super.onPause();
    google_map.onPause();
}

@Override
public void onDestroy() {
    super.onDestroy();
    google_map.onDestroy();
}

@Override
public void onLowMemory() {
    super.onLowMemory();
    google_map.onLowMemory();
}

@Override
public void onStop() {
    super.onStop();
    google_map.onStop();
}
  

Это моя ViewModel:

 private MutableLiveData<ApiResponseObject> responseLiveData = new MutableLiveData<>();
private CompositeDisposable disposables;
private ApiCallsRepository apiCallsRepository;

public MapFragmentViewModel() {
    apiCallsRepository = new ApiCallsRepository();
    disposables = new CompositeDisposable();

}

void getLocations() {

    disposables.add(apiCallsRepository.getLocations(SharedPrefUtils.getUserId())
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .doOnSubscribe(disposable -> responseLiveData.setValue(ApiResponseObject.loading()))
            .subscribe(this::setLocations, this::handleError));

}

LiveData<ApiResponseObject> locationsResponse() {
    return responseLiveData;
}

public MutableLiveData<List<LatLng>> locations = new MutableLiveData<>();

@BindingAdapter("initMap")
public static void initMap(final MapView mapView, final List<LatLng> locations) {

    if (mapView != null) {
        mapView.onCreate(new Bundle());
        mapView.getMapAsync(googleMap -> {
            if (locations != null) {
                for (LatLng latLng : locations)
                    googleMap.addMarker(new MarkerOptions().position(latLng));
            }
        });
    }
}

public void setLocations(List<LocationsResponse> mLocationsResponses) {

    List<LatLng> latLngs = new ArrayList<>();

    for (LocationsResponse locationsResponse : mLocationsResponses)
            latLngs.add(new LatLng(Double.parseDouble(locationsResponse.getLongitude()), 
                    Double.parseDouble(locationsResponse.getLatitude())));

    locations.setValue(latLngs);
    responseLiveData.setValue(ApiResponseObject.success(null));
}

private void handleError(Throwable throwable) {
    if (throwable instanceof HttpException) {
        HttpException httpException = (HttpException) throwable;
        Response<?> response = httpException.response();
        responseLiveData.setValue(ApiResponseObject.error(getErrorMessage(response)));
    } else if (throwable instanceof IOException) {
        responseLiveData.setValue(ApiResponseObject.error(UNEXPECTED_ERROR));
    } else {
        responseLiveData.setValue(ApiResponseObject.error(throwable.getMessage()));
    }
}
  

карта в xml выглядит следующим образом

 <com.google.android.gms.maps.MapView
        android:id="@ id/google_map"
        android:name="com.google.android.gms.maps.MapFragment"
        android:layout_width="0dp"
        android:layout_height="0dp"
        app:initMap="@{viewModel.locations}"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@ id/tvTextWarning" />
  

и ответ сервера

 [{
"Latitude": "19.8164793",
"Longitude": "45.2628729",
"Elevation": "128.6999969482422",
"RecordedAt": "2020-09-30T09:32:56"},{
"Latitude": "19.8164799",
"Longitude": "45.2628755",
"Elevation": "129.60000610351562",
"RecordedAt": "2020-09-30T10:35:42"},{
"Latitude": "19.8164801",
"Longitude": "45.262874",
"Elevation": "129.60000610351562",
"RecordedAt": "2020-09-30T10:34:04"}]
  

Я НАШЕЛ РЕШЕНИЕ

Проблема заключалась в том, что изменяемые данные «locations» снова устанавливались при ответном вызове, и это снова вызывало getMapAsync, и мне нужно было добавить MapView.onResume внутри этого метода.

 @BindingAdapter("initMap")
public static void initMap(final MapView mapView, final List<LatLng> locations) {

    if (mapView != null) {
        mapView.onCreate(new Bundle());
        mapView.getMapAsync(googleMap -> {
            if (locations != null) {
                mapView.onResume();
                for (LatLng latLng : locations)
                    googleMap.addMarker(new MarkerOptions().position(latLng));
            }
        });
    }
}
  

Ответ №1:

Вам нужно загрузить карту во фрагменте, а не в ViewModel. Переместите этот код в onViewCreated фрагмента

 mapView.getMapAsync(googleMap -> {
        if (locations != null) {
            for (LatLng latLng : locations)
                googleMap.addMarker(new MarkerOptions().position(latLng));
        }
    });
  

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

1. Спасибо за ответ. Проблема не в том, что я загружаю карту, я нашел решение. Я отредактировал ответ, поэтому проверьте его, если вам интересно.

2. @VladaMomirovic Хорошо. Но я думаю, что обратный вызов getMapAsync должен быть отправной точкой для вызова getLocations, это гораздо более понятный способ