Ошибка открытия SupportMapFragment во второй раз

При открытии моего SupportMapFragment (Android maps v2) во второй раз (вызов setContentView) я получаю следующую ошибку:

01-28 16:27:21.374: E/AndroidRuntime(32743): FATAL EXCEPTION: main
01-28 16:27:21.374: E/AndroidRuntime(32743): android.view.InflateException: Binary XML file line #6: Error inflating class fragment
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:704)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.rInflate(LayoutInflater.java:746)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.inflate(LayoutInflater.java:489)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.inflate(LayoutInflater.java:396)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.inflate(LayoutInflater.java:352)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.View.inflate(View.java:16119)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at mypackage.MyView.<init>(HitsView.java:26)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at mypackage.MenuListFragment.onItemClick(MenuListFragment.java:133)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.widget.AdapterView.performItemClick(AdapterView.java:298)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.widget.AbsListView.performItemClick(AbsListView.java:1086)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.widget.AbsListView$PerformClick.run(AbsListView.java:2855)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.widget.AbsListView$1.run(AbsListView.java:3529)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.os.Handler.handleCallback(Handler.java:615)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.os.Handler.dispatchMessage(Handler.java:92)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.os.Looper.loop(Looper.java:137)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.app.ActivityThread.main(ActivityThread.java:4745)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at java.lang.reflect.Method.invokeNative(Native Method)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at java.lang.reflect.Method.invoke(Method.java:511)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at dalvik.system.NativeStart.main(Native Method)
01-28 16:27:21.374: E/AndroidRuntime(32743): Caused by: java.lang.IllegalArgumentException: Binary XML file line #6: Duplicate id 0x7f04003b, tag null, or parent id 0x0 with another fragment for com.google.android.gms.maps.SupportMapFragment
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.support.v4.app.FragmentActivity.onCreateView(FragmentActivity.java:285)
01-28 16:27:21.374: E/AndroidRuntime(32743):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:676)
01-28 16:27:21.374: E/AndroidRuntime(32743):    ... 20 more

XML-файл:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:map="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >
    <fragment
        android:id="@+id/hits_map"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        class="com.google.android.gms.maps.SupportMapFragment"
        map:mapType="normal"
        map:uiZoomControls="false"
        map:uiZoomGestures="true"
        map:cameraZoom="13"
        map:uiRotateGestures="true"
        map:uiTiltGestures="true"/>
</RelativeLayout>

MyView.класс:

public class MyView extends RelativeLayout {
    private GoogleMap map;

    public MyView(Context context, FragmentActivity activity) {
        super(context);
        inflate(activity, R.layout.activity_hits, this);
        this.map = ((SupportMapFragment) activity.getSupportFragmentManager()
            .findFragmentById(R.id.hits_map)).getMap();
    }
}

Я понятия не имею, что означает эта ошибка. Кто-нибудь может это объяснить?


person Klaasvaak    schedule 28.01.2013    source источник
comment
У вас есть View, пытающийся раздуть макет, содержащий фрагмент? Если да, то почему?   -  person CommonsWare    schedule 28.01.2013
comment
Я. Это не правильный способ его использования? Я использую слайдингменю. В своих документах они говорят, что нужно вызвать setContentView, чтобы отобразить основной вид. А основным видом в данном случае является вид, содержащий фрагмент. ссылка: github.com/jfeinstein10/SlidingMenu#simple-example   -  person Klaasvaak    schedule 28.01.2013
comment
Я бы точно никогда так не поступил.   -  person CommonsWare    schedule 28.01.2013
comment
Вы в конце концов решили это?   -  person Thomas Kaliakos    schedule 10.02.2013
comment
Вместо этого я использовал MapView, я на самом деле не знал, что MapView тоже существует.   -  person Klaasvaak    schedule 14.02.2013
comment
[Это ссылка, которая поможет вам, у меня возникла та же проблема, но это сработало для меня :)] [1] [1]: stackoverflow.com/questions /14083950/   -  person Trino EGlez    schedule 24.08.2013
comment
Я думаю, что это лучшее решение: stackoverflow.com/a/19128882/1932352   -  person jimbo82    schedule 01.11.2013


Ответы (10)


Вместо объявления de SupportMapFragment в макете сделайте это программно и убедитесь, что вы используете getChildFragmentMananger вместо классического getFragmentManager() для создания фрагмента.

  mMapFragment = SupportMapFragment.newInstance();
    FragmentTransaction fragmentTransaction =
             mMapFragment.getChildFragmentManager().beginTransaction();
     fragmentTransaction.add(R.id.map_root, mMapFragment);
     fragmentTransaction.commit(); 

Сохраните этот SupportMapFragment mMapFragment, так как он понадобится вам для получения объекта GoogleMap:

  GoogleMap map = mMapFragment.getMap();
person juanmeanwhile    schedule 04.02.2013
comment
Да, вам нужно дождаться запуска, прежде чем карта будет инициализирована. Я использую обработчик задержки, чтобы получить ссылку на карту через 10 миллисекунд, или же использовать жизненный цикл в своих интересах. то есть зафиксировать фрагмент в onCreate, а затем получить ссылку на карту в onResume. - person Aiden Fry; 18.06.2014
comment
@juanmeanwhile, вместо этого вы должны использовать getMapAsync(). MapFragment - person Gabriel Kaffka; 26.03.2015

Обновление: В качестве альтернативного решения (которое я считаю намного лучше) вы можете использовать MapView, который описан: здесь

Я столкнулся с похожей проблемой при работе с реализацией вкладок. В Google Maps V2 вы застряли с SupportMapFragment, поэтому использование MapView невозможно. Между сообщением juanmeanwhile и комментарием №1, найденным здесь (https://code.google.com/p/gmaps-api-issues/issues/detail?id=5064#c1), мне удалось выяснить, что я делаю неправильно.

Итак, для всех, кто получает ошибку Duplicate id, убедитесь, что вы объявляете фрагмент программно, а не в XML. Это, вероятно, означает вложенные макеты.

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:map="http://schemas.android.com/apk/res-auto"
   android:layout_width="match_parent"
   android:layout_height="match_parent" >

<!-- Lots of fancy layout -->   

<RelativeLayout
        android:id="@+id/map"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
    </RelativeLayout>
</RelativeLayout>

Затем вам нужно создать свой фрагмент программно, но это нужно сделать тщательно с учетом жизненного цикла фрагментов (http://developer.android.com/reference/android/app/Fragment.html#getChildFragmentManager()). Чтобы все было создано в нужное время, ваш код должен выглядеть так (взято из комментария №1).

public class MyFragment extends Fragment {

private SupportMapFragment fragment;
private GoogleMap map;

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

@Override
public void onActivityCreated(Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);
    FragmentManager fm = getChildFragmentManager();
    fragment = (SupportMapFragment) fm.findFragmentById(R.id.map);
    if (fragment == null) {
        fragment = SupportMapFragment.newInstance();
        fm.beginTransaction().replace(R.id.map, fragment).commit();
    }
}

@Override
public void onResume() {
    super.onResume();
    if (map == null) {
        map = fragment.getMap();
        map.addMarker(new MarkerOptions().position(new LatLng(0, 0)));
    }
}
}

Надеюсь, это сэкономит время.

person Kaleb    schedule 04.04.2013
comment
Сохранит ли это экземпляр карты во всех фрагментах? Или он будет уничтожен после onPause()? - person Ben Sewards; 07.07.2013
comment
К вашему сведению, вы не застряли с SupportMapFragment, MapView все еще существует и полностью функционален: developers.google.com/maps/documentation/android/reference/com/ - person Murphy; 31.07.2013
comment
Я думаю, если вы используете Fragment, вы на самом деле застряли с MapFragment и действительно поддерживаете MapFragment, если хотите поддерживать Honeycomb. Если вы используете Activity, это не так, и вы можете использовать MapView (хотя лично я этого не подтверждал). Итак, в моем примере вы в основном застряли с этой реализацией. Если бы вы использовали действие, вы могли бы использовать MapView, но вам все равно нужно было бы управлять жизненным циклом. - person Kaleb; 31.07.2013
comment
Это не решает проблему, оно все равно вылетает, когда я запускаю его снова. - person alicanbatur; 20.09.2013
comment
@ Калеб, да, но я это исправил. Я пробовал много вещей, и, наконец, я решаю это. Но я действительно не понимаю, как я это исправил. - person alicanbatur; 20.09.2013
comment
@Kaleb, спасибо за идею. Но на самом деле, я думаю, Google ненавидит всех нас. - person jimpanzer; 02.10.2013
comment
Код генерирует ошибку, которая была решена здесь: #15656428" title="получение сообщения об ошибке java lang absolutestateException был уничтожен"> stackoverflow.com/questions/15207305/ (ответ Маркуса) - person jiahao; 15.11.2013
comment
После безумного поиска в Интернете это решение сработало для меня. Спасибо! - person Marcelo; 17.09.2014
comment
Это также работает для меня, и я думаю, что лучше всего, основываясь на ответах, которые я видел сам, которые не требуют MapView. - person gorbos; 18.12.2014

Я потратил полдня на решение этой проблемы и нашел только обходной путь. Переопределите свой метод onCreate в классе YourFragment:

public void onCreate(Bundle savedInstanceState) {
    setRetainInstance(true); 
    super.onCreate(savedInstanceState);     
}

И переопределите ваш метод onDestroy:

@Override
public void onDestroyView() {
    super.onDestroyView();
    try {
        SupportMapFragment fragment = (SupportMapFragment) getActivity()
                                          .getSupportFragmentManager().findFragmentById(
                                              R.id.multi_inns_on_map);
        if (fragment != null) getFragmentManager().beginTransaction().remove(fragment).commit();

    } catch (IllegalStateException e) {
        //handle this situation because you are necessary will get 
        //an exception here :-(
    }
}
person Jaroslav    schedule 19.09.2013
comment
Этот ответ позволяет вам определить ваши дочерние фрагменты в макетах XML, которые я считаю более управляемыми, чем программная замена фрагментов макета фрагментами. - person Valerio Santinelli; 03.10.2013
comment
пожалуйста, обратите внимание на блок обработки исключений, потому что вы рискуете потерять состояние объекта!! Вот почему это называется обходным путем - person Jaroslav; 14.01.2014
comment
вместо этого, если я использую блок try-catch if (f != null && f.isResumed()) {}, он работает для меня - person Analizer; 06.07.2014
comment
@Jaroslav: На каком этапе я получу исключение? - person TheDevMan; 12.09.2014
comment
@TheDevMan - например, при повороте экрана } catch (IllegalStateException e) { // обработайте эту ситуацию, потому что вы должны получить // здесь исключение :-( } - person Jaroslav; 25.12.2014
comment
Пожалуйста, используйте этот подход для решения проблемы, описанной выше. И не забудьте удалить setRetainInstance(true) в методе onCreate code.google.com/p/gmaps-api-issues/issues/detail?id=5064#c1 - person Jaroslav; 26.01.2015
comment
Это сработало для меня! Спасибо, это должен быть правильный ответ. - person ButterBeast; 11.03.2015

Я провел весь этот день в поисках решения этой проблемы, прочитав все обходные пути здесь, и я нашел этот способ решения проблем. Я опубликую весь код, потому что на этот вопрос нет полного ответа, только небольшой фрагмент, который соответствует полному сценарию и помогает всем.

public class LocationFragment extends Fragment implements View.OnClickListener {

    private GoogleMap googleMap;
    private static View rootView;
    private SupportMapFragment supportMapFragment;

    public LocationFragment() {
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
    }

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

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        if(savedInstanceState == null) {

            if (rootView != null) {
                ViewGroup parent = (ViewGroup) rootView.getParent();
                if (parent != null)
                    parent.removeView(rootView);
            }

            try{
                if(rootView == null)
                {
                    rootView = inflater.inflate(R.layout.fragment_location, container, false);
                }

                supportMapFragment = (SupportMapFragment) getFragmentManager().findFragmentById(R.id.map);
                LocationUtility locationUtility = LocationUtility.getInstance(context);
                GoogleMap googleMap =  locationUtility.initilizeMap(supportMapFragment);

              //More code

            } catch (Exception e)
            {
                e.printStackTrace();
            }
        }

        return rootView;
    }

}

Выполняя отладку, вы заметите, что не имеет значения, как работает фрагмент, вам не нужен botter для onDestroyMethod и т. д., поскольку проверка нулевого значения просто работает с текущим экземпляром объекта.

Наслаждаться! ;)

person Sterling Diaz    schedule 29.08.2014
comment
Огромное спасибо!! Я застрял с этой проблемой больше недели. Хотя я использую Xamarin Android (C#), но ваше решение сработало отлично! - person Robert; 05.01.2016

Я столкнулся с этой проблемой и обнаружил в журналах Android, что мое аппаратное ускорение не включено. После того, как я включил его в своем AndroidManifest, я показывал карты несколько раз, и это больше не было проблемой. Я использовал эмулятор x86.

person user2771209    schedule 12.09.2013

После множества хитов и попыток мне, наконец, удалось идеально использовать фрагмент MapView в моем приложении, и результат выглядит следующим образом: -

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

, вот мой класс фрагментов MapView: -

import android.content.Context;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Toast;

import com.google.android.gms.maps.CameraUpdateFactory;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.GoogleMap.OnMapClickListener;
import com.google.android.gms.maps.MapFragment;
import com.google.android.gms.maps.model.BitmapDescriptorFactory;
import com.google.android.gms.maps.model.CameraPosition;
import com.google.android.gms.maps.model.LatLng;
import com.google.android.gms.maps.model.MarkerOptions;
import com.serveroverload.yago.R;


public class HomeFragment extends Fragment implements LocationListener {
    // Class to do operations on the Map
    GoogleMap googleMap;
    private LocationManager locationManager;

    public static Fragment newInstance() {
        return new HomeFragment();
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.home_fragment, container, false);
        Bundle bdl = getArguments();

        // setuping locatiomanager to perfrom location related operations
        locationManager = (LocationManager) getActivity().getSystemService(
                Context.LOCATION_SERVICE);

        // Requesting locationmanager for location updates
        locationManager.requestLocationUpdates(
                LocationManager.NETWORK_PROVIDER, 1, 1, this);

        // To get map from MapFragment from layout
        googleMap = ((MapFragment) getActivity().getFragmentManager()
                .findFragmentById(R.id.map)).getMap();

        // To change the map type to Satellite
        // googleMap.setMapType(GoogleMap.MAP_TYPE_SATELLITE);

        // To show our current location in the map with dot
        // googleMap.setMyLocationEnabled(true);

        // To listen action whenever we click on the map
        googleMap.setOnMapClickListener(new OnMapClickListener() {

            @Override
            public void onMapClick(LatLng latLng) {
                /*
                 * LatLng:Class will give us selected position lattigude and
                 * longitude values
                 */
                Toast.makeText(getActivity(), latLng.toString(),
                        Toast.LENGTH_LONG).show();
            }
        });

        changeMapMode(3);

        // googleMap.setSatellite(true);
        googleMap.setTrafficEnabled(true);
        googleMap.setBuildingsEnabled(true);
        googleMap.setMyLocationEnabled(true);

        return v;
    }

    private void doZoom() {
        if (googleMap != null) {
            googleMap.animateCamera(CameraUpdateFactory.newLatLngZoom(
                    new LatLng(18.520430, 73.856744), 17));
        }
    }

    private void changeMapMode(int mapMode) {

        if (googleMap != null) {
            switch (mapMode) {
            case 0:
                googleMap.setMapType(GoogleMap.MAP_TYPE_NONE);
                break;

            case 1:
                googleMap.setMapType(GoogleMap.MAP_TYPE_NORMAL);
                break;

            case 2:
                googleMap.setMapType(GoogleMap.MAP_TYPE_SATELLITE);
                break;

            case 3:
                googleMap.setMapType(GoogleMap.MAP_TYPE_TERRAIN);
                break;

            case 4:
                googleMap.setMapType(GoogleMap.MAP_TYPE_HYBRID);
                break;

            default:
                break;
            }
        }
    }

    private void createMarker(double latitude, double longitude) {
        // double latitude = 17.385044;
        // double longitude = 78.486671;

        // lets place some 10 random markers
        for (int i = 0; i < 10; i++) {
            // random latitude and logitude
            double[] randomLocation = createRandLocation(latitude, longitude);

            // Adding a marker
            MarkerOptions marker = new MarkerOptions().position(
                    new LatLng(randomLocation[0], randomLocation[1])).title(
                    "Hello Maps " + i);

            Log.e("Random", "> " + randomLocation[0] + ", " + randomLocation[1]);

            // changing marker color
            if (i == 0)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_AZURE));
            if (i == 1)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_BLUE));
            if (i == 2)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_CYAN));
            if (i == 3)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_GREEN));
            if (i == 4)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_MAGENTA));
            if (i == 5)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_ORANGE));
            if (i == 6)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_RED));
            if (i == 7)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_ROSE));
            if (i == 8)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_VIOLET));
            if (i == 9)
                marker.icon(BitmapDescriptorFactory
                        .defaultMarker(BitmapDescriptorFactory.HUE_YELLOW));

            googleMap.addMarker(marker);

            // Move the camera to last position with a zoom level
            if (i == 9) {
                CameraPosition cameraPosition = new CameraPosition.Builder()
                        .target(new LatLng(randomLocation[0], randomLocation[1]))
                        .zoom(15).build();

                googleMap.animateCamera(CameraUpdateFactory
                        .newCameraPosition(cameraPosition));
            }
        }

    }

    /*
     * creating random postion around a location for testing purpose only
     */
    private double[] createRandLocation(double latitude, double longitude) {

        return new double[] { latitude + ((Math.random() - 0.5) / 500),
                longitude + ((Math.random() - 0.5) / 500),
                150 + ((Math.random() - 0.5) * 10) };
    }

    @Override
    public void onLocationChanged(Location location) {

        if (null != googleMap) {
            // To get lattitude value from location object
            double latti = location.getLatitude();
            // To get longitude value from location object
            double longi = location.getLongitude();

            // To hold lattitude and longitude values
            LatLng position = new LatLng(latti, longi);

            createMarker(latti, longi);

            // Creating object to pass our current location to the map
            MarkerOptions markerOptions = new MarkerOptions();
            // To store current location in the markeroptions object
            markerOptions.position(position);

            // Zooming to our current location with zoom level 17.0f
            googleMap.animateCamera(CameraUpdateFactory.newLatLngZoom(position,
                    17f));

            // adding markeroptions class object to the map to show our current
            // location in the map with help of default marker
            googleMap.addMarker(markerOptions);
        }

    }

    @Override
    public void onStatusChanged(String provider, int status, Bundle extras) {
        // TODO Auto-generated method stub

    }

    @Override
    public void onProviderEnabled(String provider) {
        // TODO Auto-generated method stub

    }

    @Override
    public void onProviderDisabled(String provider) {
        // TODO Auto-generated method stub

    }

    @Override
    public void onDestroyView() {
        // TODO Auto-generated method stub
        super.onDestroyView();

        locationManager.removeUpdates(this);

        android.app.Fragment fragment = getActivity().getFragmentManager()
                .findFragmentById(R.id.map);
        if (null != fragment) {
            android.app.FragmentTransaction ft = getActivity()
                    .getFragmentManager().beginTransaction();
            ft.remove(fragment);
            ft.commit();
        }
    }


}

Мой файл Xml выглядит так: -

<?xml version="1.0" encoding="utf-8"?>
<fragment xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/map"
    android:name="com.google.android.gms.maps.MapFragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

Самое важное, что следует отметить, это то, что НЕ смешивайте app.Fragment с v4.Fragments, иначе приложение будет аварийно завершать работу.

Как видите, я использовал app.Fragment для присоединения и удаления своего фрагмента MapView.

Надеюсь, это поможет кому-то

person Hitesh Sahu    schedule 18.12.2015
comment
Удалось ли вам добавить кнопку «Мое местоположение» в mapView? а также подсвечивать текущее местоположение пользователя в режиме реального времени? - person voghDev; 19.08.2016

Я пробовал различные решения, предложенные здесь, и лучшим, ИМХО, является MapView. Во всяком случае, я нашел другое решение, которое может удовлетворить ваши потребности (а может и нет).

У меня был SupportMapFragment внутри ViewPager+FragmentPagerAdapter с 3 вкладками. Поэтому я просто настроил ViewPager таким образом, чтобы он никогда не уничтожал вкладку, где находится SupportMapFragment.

viewPager.setOffScreenPageLimit(2);
viewPager.setAdapter(...);

С помощью этого вызова ViewPager всегда будет поддерживать две страницы после и две страницы перед текущей, поэтому ни одна из моих трех вкладок никогда не уничтожается/не воссоздается.

Задача решена :)

Обратите внимание, что потребление памяти увеличится

person voghDev    schedule 19.08.2016
comment
большое спасибо. У меня есть просмотр пейджера с 3 фрагментами, и это было именно то, что мне нужно - person itzhar; 17.05.2017

попробуйте этот код

  @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
         finder = new LocationFinder(getActivity());

             view = inflater.inflate(R.layout.fragment_secondfragement, container, false);



                        initMap();

        
      //  getLocation();
        return view;

    }


    public void initMap(){

        if (map == null) {
            mapFragment = (SupportMapFragment) getChildFragmentManager()
                    .findFragmentById(R.id.map);
        }

        map = mapFragment.getMap();
        map.setMyLocationEnabled(true);
        map.getUiSettings().setMyLocationButtonEnabled(true);

    }

person Ameen Maheen    schedule 09.01.2015

У меня такая же проблема, и она решена путем динамического создания карты Google. Вам нужно создать класс, расширяющий класс SupportMapFragment.

Этот метод работает с имбирным пряником в леденец и с вложенными фрагментами, потому что он создаст карту динамически, поэтому не нужно беспокоиться об удалении фрагмента или чего-то подобного.

Шаг 1:

public class DynamicMapFragment extends SupportMapFragment {

    public GoogleMap googleMap;
    public View mOriginalContentView;
    public static int statusGooglePlayService;    

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

    @Override
    public View onCreateView(LayoutInflater mInflater, ViewGroup parent,
            Bundle arg2) {

        mOriginalContentView = super.onCreateView(mInflater, parent, arg2);

       //this is for getting height of the phone screen
        DisplayMetrics displaymetrics = new DisplayMetrics();
        getActivity().getWindowManager().getDefaultDisplay()
                .getMetrics(displaymetrics);
        int height = displaymetrics.heightPixels;

        //Here you can define the height of the map as you desire
        FrameLayout.LayoutParams parentLayout = new FrameLayout.LayoutParams(
                FrameLayout.LayoutParams.MATCH_PARENT, height / 3);

        mOriginalContentView.setLayoutParams(parentLayout);
        return mOriginalContentView;
    }

    @Override
    public View getView() {
        return mOriginalContentView;
    }

    @Override
    public void onInflate(Activity arg0, AttributeSet arg1, Bundle arg2) {
        super.onInflate(arg0, arg1, arg2);
    }

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        googleMap = getMap();

        statusGooglePlayService = GooglePlayServicesUtil
                .isGooglePlayServicesAvailable(getActivity());
        if (statusGooglePlayService == ConnectionResult.SUCCESS) {
            if (googleMap != null) {
                // do whatever you want to do with map
            }
        } else {
            Dialog dialog = GooglePlayServicesUtil.getErrorDialog(
                    statusGooglePlayService, getActivity(), -1);
            dialog.show();
        }
    }

}

Затем вызовите созданный выше класс из Activity, Fragment следующим образом:

Шаг 2:

//Calling the method from a fragment
private void createMap(View rootView) {
        FrameLayout mapLayout = (FrameLayout)rootView.findViewById(R.id.location_map);
        FragmentTransaction mTransaction = getActivity()
                .getSupportFragmentManager().beginTransaction();
        SupportMapFragment mFRaFragment = new DynamicMapFragment();
        mTransaction.add(mapLayout.getId(), mFRaFragment);
        mTransaction.commit();

        try {
            MapsInitializer.initialize(activity);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

XML-макет для фрагмента, где необходимо добавить Framelayout вместо фрагмента для карты

Шаг 3.

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >



    <FrameLayout
        android:id="@+id/location_map"
        android:layout_width="match_parent"
        android:layout_height="250dp"
        android:layout_weight="0" />

      <!--  You can add any other view also whatever you want to add 

      <Button
        android:layout_width="match_parent"
        android:layout_height=""wrap_content""
        android:layout_weight="0" />    -->

</LinearLayout>
person Pankaj    schedule 14.08.2015
comment
Привет, не могли бы вы взглянуть на мою проблему, она похожа на то, что вы объяснили, но у меня не работает. будет большой милостью. Вот ссылка: - stackoverflow.com/questions/32240138/ - person Nicks; 05.09.2015

просто наденьте onCreateView

if (view != null) {
        ViewGroup parent = (ViewGroup) view.getParent();
        if (parent != null)
            parent.removeView(view);
    }

view = inflater.inflate(R.layout.map, container, false);
person Kastriot Dreshaj    schedule 03.11.2015