Загрузка фрагмента Android Google Map, управляемого ViewPager

Я могу загрузить карту Google во фрагмент Android, который находится в действии. Это работает нормально.

Но теперь я хочу использовать ViewPager для навигации между представлениями (класс android.support.v4.app.Fragment). Не представляется возможным загрузить com.google.android.gms.maps.MapFragment в такой фрагмент.

Например, в:

SoleMap.java

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import com.google.android.gms.maps.CameraUpdateFactory;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.MapFragment;
import com.google.android.gms.maps.OnMapReadyCallback;
import com.google.android.gms.maps.model.LatLng;

public class SoleMap extends Fragment implements OnMapReadyCallback {

    MapFragment gMapFragment;
    GoogleMap gMap = null;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.sole_map, container, false);

        gMapFragment = (MapFragment) getFragmentManager().findFragmentById(R.id.soleViewMap);
        gMapFragment.getMapAsync(this);

        return view;
    }

    @Override
    public void onMapReady(GoogleMap map) {
        gMap = map;
        gMap.setMapType(GoogleMap.MAP_TYPE_HYBRID);
        gMap.moveCamera(CameraUpdateFactory.newLatLngZoom(new
                LatLng(49.39,-124.83), 20));
    }

}

Заявление

getFragmentManager().findFragmentById(R.id.holeViewMap);

вызывает ошибку компилятора (несовместимые типы).

Вместо этого я попытался использовать SupportMapFragment, который устраняет ошибки компилятора, но затем, когда я запускаю приложение, оно немедленно завершается с сообщением «Ошибка ввода-вывода: отказано в соединении». Документы Google, похоже, указывают на то, что вам нужна специальная учетная запись «Для работы», чтобы использовать библиотеку поддержки. Это правильно? Если да, то я думаю, что мне не повезло.

Единственный другой способ, который я вижу, — это использовать действия вместо фрагментов для размещения моих представлений, т. е. избавиться от ViewPager.

Какие-либо предложения?


person NLRicker    schedule 30.04.2015    source источник
comment
попробуйте это: stackoverflow.com/questions/19353255/   -  person Josef    schedule 30.04.2015
comment
Соединение отклонено: у вас есть ключ API Карт Google в манифесте?   -  person Eugen Pechanec    schedule 01.05.2015
comment
Я думаю, что моя проблема с отказом в соединении не была связана - вероятно, связана с обновлением SDK, которое требовало выключения/перезагрузки. Чтобы ответить на ваш вопрос, да, у меня был мой манифест (с ключом), который работал ранее.   -  person NLRicker    schedule 01.05.2015


Ответы (5)


В вашем sole_map.xml вы должны добавить MapView, скажем, что-то вроде этого:

<?xml version="1.0" encoding="utf-8"?>
<com.google.android.gms.maps.MapView
    android:id="@+id/soleViewMap"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

Затем получите его из фрагмента SoleMap с идентификатором view.findViewById(R.id.soleViewMap);. Это вернет вам MapView, после чего вы сможете делать то, что делаете сейчас. Это должно выглядеть примерно так:

public class SoleMap extends Fragment implements OnMapReadyCallback {

  MapView gMapView;
  GoogleMap gMap = null;

  @Override
  public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.sole_map, container, false);

    gMapView = (MapView) view.findViewById(R.id.soleViewMap);
    gMapView.getMapAsync(this);

    return view;
  }

  @Override
  public void onMapReady(GoogleMap map) {
    gMap = map;
    gMap.setMapType(GoogleMap.MAP_TYPE_HYBRID);
    gMap.moveCamera(CameraUpdateFactory.newLatLngZoom(new
            LatLng(49.39,-124.83), 20));
  }
}

Не забудьте связать свое представление карты с жизненным циклом фрагмента, вызывая события onCreate, onLowMemory, onPause, onResume и onDestroy (надеюсь, не пропустите ни одно из них), а также вызовите MapsInitializer.initialize(getContext());

person Jorge Mendez    schedule 30.04.2015
comment
Я все еще работаю над реализацией этого, но на случай, если кто-то еще использует вышеизложенное, я считаю, что это требует одного исправления. Вместо findFragmentById() нужно использовать findViewByID(). Затем я могу скомпилировать. Я вернусь как можно скорее, чтобы указать, смогу ли я заставить его работать таким образом. - person NLRicker; 01.05.2015
comment
Ты прав, мой плохой, я сейчас внесу поправку. Спасибо! - person Jorge Mendez; 01.05.2015
comment
Теперь я могу получить карту на одном из своих фрагментов. Вышеизложенное было очень полезно в отношении общего исправления. Справку по деталям реализации см. на странице gist.github.com/joshdholtz/4522551. Также документы Android для класса MapView. - person NLRicker; 01.05.2015
comment
Почему MapView, а не SupportMapFragment? - person IgorGanapolsky; 17.09.2015
comment
Благодаря тонну ! этот пост должен иметь +100, все просто не забудьте вызвать mMapView.onCreate() в createView и mMapView.onResume() в обратном вызове - person Tintinabulator Zea; 24.12.2015
comment
Это сработало для меня. Я пропустил gMapView.getMapAsync(this); в своем проекте. - person Hasan Abdullah; 11.01.2017

Принятый ответ - хорошая отправная точка, но если мы попытаемся его использовать, он потерпит неудачу, он не описывает полное решение. Полный рабочий фрагмент с картой, которую можно использовать в ViewPager и Tabs (обратите внимание на использование карты в жизненном цикле фрагментов):

public class AttractionMapTabFragment extends AttractionTab implements OnMapReadyCallback {

private ScrollMapView gMapView;
private GoogleMap gMap;


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

    View view = inflater.inflate(R.layout.fragment_map, null);

    gMapView = (ScrollMapView) view.findViewById(R.id.map);
    gMapView.getMapAsync(this);

    //most important use onCreate;
    gMapView.onCreate(getArguments());
    return view;

}

@Override
public void onMapReady(GoogleMap googleMap) {
    gMap = googleMap;
    gMap.moveCamera(CameraUpdateFactory.newLatLngZoom(new
            LatLng(51.3623329,21.7719342), 8));
}

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

    if (gMapView != null)
        gMapView.onResume();
}


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

    if (gMapView != null)
        gMapView.onDestroy();
}

@Override
public void onStart() {
    super.onStart();

    if (gMapView != null)
        gMapView.onStart();
}

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

    if (gMapView != null)
        gMapView.onStop();

}

@Override
public void onSaveInstanceState(Bundle outState) {
    super.onSaveInstanceState(outState);

    if (gMapView != null)
        gMapView.onSaveInstanceState(outState);
}
}

Пользовательский класс MapView - он необходим, если мы используем ScrollView в качестве родителя, он дает возможность перемещать карту. Если вы не используете ScrollView на этом экране, то можно использовать стандартный MapView.

public class ScrollMapView extends MapView {

public ScrollMapView(Context context, AttributeSet attributeSet) {
    super(context, attributeSet);
}

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
    /**
     * Request all parents to relinquish the touch events
     */
    getParent().requestDisallowInterceptTouchEvent(true);
    return super.dispatchTouchEvent(ev);
}
}

Последнее, что касается макета - R.layout.fragment_map:

<?xml version="1.0" encoding="utf-8"?>
<your.app.ScrollMapView
android:id="@+id/map"
android:layout_width="match_parent"
android:layout_height="match_parent"
xmlns:android="http://schemas.android.com/apk/res/android" />
person Maciej Sikora    schedule 09.01.2017

Запишите этот код в свой activity_maps.xml

<fragment xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:map="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/map"
    android:name="com.google.android.gms.maps.SupportMapFragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    />
person Rahul Bhardwaj    schedule 24.10.2016

Я застрял с этой проблемой в течение нескольких дней. Наконец, я заставил это работать так:

единственная_карта.xml

<fragment xmlns:map="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    tools:context=".SoleMap"
    android:id="@+id/soleViewMap"
    android:name="com.google.android.gms.maps.SupportMapFragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    map:mapType="terrain" />

SoleMap.java

Вместо того, чтобы делать что-то в «onCreateView», я делаю это в «onViewCreated», а также вместо использования MapFragment я использую SupportMapFragment и использую getChildFragmentManager вместо getFragmentManager, например:

public class MapaFragment extends Fragment implements OnMapReadyCallback {

GoogleMap mapa_sole;

public View onCreateView(@NonNull LayoutInflater inflater,
                         ViewGroup container, Bundle savedInstanceState) {
    return inflater.inflate(R.layout.sole_map, container, false);
}

@Override
public void onViewCreated(View view, Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);

    final SupportMapFragment mapFragment = (SupportMapFragment) getChildFragmentManager().findFragmentById(R.id.soleViewMap);
    mapFragment.getMapAsync(this);

}

@Override
public void onMapReady(GoogleMap googleMap) {
    mapa_sole = googleMap;


    // TODO....

   }
}

Надеюсь, это поможет!

person Manuela Viola    schedule 30.11.2019

person    schedule
comment
Вы ничего не объяснили, каково ваше решение? Это просто автоматически сгенерированный код при создании MapActivity из шаблонов Android. - person IgniteCoders; 01.05.2019
comment
Наконец я нашел то, что вы хотите решить с помощью этого кода, он заставляет мою карту работать внутри пейджера просмотра. Но было трудно понять, что предлагает ваш код/решение. - person IgniteCoders; 01.05.2019