【发布时间】:2015-06-09 21:48:11
【问题描述】:
我正在使用基于地图的应用程序,该应用程序创建包含地图和 Web 视图的相册视图。该问题发生在轮换期间,并且在 com.android.support:support-v4:19.1 上运行良好,但在我更新到 22.2.0 后无法正常工作。
地图在初始视图中按预期显示,但在旋转后,地图不再出现。我验证了 GoogleMap、容器视图(保存地图)和 SupportMapFragment 都是有效的。
在调试时,我决定检查 SupportMapFragment 的视图,看看这是否可能是原因。旋转后,对 SupportMapFragment 上的 getView() 的调用返回 null (我怀疑这就是地图没有出现的原因,但我不明白为什么它最初有效并且旋转后为 null)。
我不知所措,因为据我所知,我有一个有效的 SupportMapFragment、一个有效的 GoogleMap 和一个具有适当大小(在本例中为 680x800)的容器视图。正如我上面所说,在我使用 19.1 之前的版本中,此代码按预期工作。
下面是相关的sn-ps代码:
SupportMapManager 创建在 EventReportAlbumActivity 类中
public class EventReportAlbumActivity extends FragmentActivity
{
@Nullable protected SupportMapFragment m_mapFragment;
@NotNull
public SupportMapFragment getMapFragment ()
{
if( m_mapFragment == null )
{
GoogleMapOptions options = new GoogleMapOptions();
options.compassEnabled( false );
options.rotateGesturesEnabled( false );
options.scrollGesturesEnabled( false );
options.tiltGesturesEnabled( false );
options.zoomControlsEnabled( false );
options.zoomGesturesEnabled( false );
m_mapFragment = SupportMapFragment.newInstance( options );
m_mapFragment.setRetainInstance( true );
}
return m_mapFragment;
}
}
显示地图和 web 视图的实际 Fragment 实例
public class EventAlbumItemFragment extends Fragment
{
@Nullable
protected SupportMapFragment m_mapFragment;
public static EventAlbumItemFragment newInstance ( )
{
final EventAlbumItemFragment fragment = new EventAlbumItemFragment();
fragment.setRetainInstance( true );
return fragment;
}
@Override
public View onCreateView ( @NotNull LayoutInflater inflater,
@NotNull ViewGroup container,
@Nullable Bundle savedInstanceState )
{
final View view = inflater.inflate( R.layout.event_album_item, container, false );
View mapContainer = view.findViewById( R.id.mapview_container );
// make map fragment view invisible until we configure it so the coordinates will be correct
mapContainer.setVisibility( View.INVISIBLE );
// other code removed for brevity...
}
/**
* Setter for the map fragment and its UI
*
* @param mapFragment The map fragment which should be added to this fragment's view, or null to remove the current fragment
*/
public void setMapFragment ( @Nullable SupportMapFragment mapFragment )
{
final FragmentManager fragmentManager = getChildFragmentManager();
if( m_mapFragment != null && m_mapFragment.isAdded() )
{
// Remove the existing map manager
fragmentManager.beginTransaction()
.remove( m_mapFragment )
.commit();
}
if( mapFragment != null )
{
// Add the map view to the current view
fragmentManager.beginTransaction()
.add( R.id.mapview_container, mapFragment, MAP_FRAGMENT_TAG )
.commit();
}
fragmentManager.executePendingTransactions();
m_mapFragment = mapFragment;
configureMap();
}
protected void configureMap ()
{
if ( m_mapFragment != null && m_eventMapFeature != null && m_eventMapFeature.haveValidLocation() == true )
{
final EventAlbumItemFragment localThis = this ;
m_mapFragment.getMapAsync( new OnMapReadyCallback()
{
@Override
public void onMapReady( GoogleMap googleMap )
{
final View mapContainer = getView().findViewById( R.id.mapview_container );
final int mapWidth = mapContainer.getWidth();
final int mapHeight = mapContainer.getHeight();
// we have our map and the coordinates should be correct. Make map visible and proceed
mapContainer.setVisibility( View.VISIBLE );
// This line below crashes because view is NULL after rotation.
// Left here to illustrate what I believe to be the root cause of the issue
// m_mapFragment.getView().setVisibility( View.VISIBLE );
// Remove all overlays from the map
googleMap.clear();
final DisplayMetrics displayMetrics = getResources().getDisplayMetrics();
final float zoom = (float)GeoUtils.getZoomForMetersWide( MAP_MINIMUM_METER_SPAN,
mapWidth / displayMetrics.scaledDensity,
m_eventMapFeature.getCoordinate().latitude );
// Change the map bounds
CameraUpdate cameraUpdate;
// draw some items on the map
final LatLngBounds bounds = m_eventMapFeature.getBounds();
cameraUpdate = CameraUpdateFactory.newLatLngZoom( bounds.southwest, zoom );
googleMap.moveCamera( cameraUpdate );
}
} );
}
}
/** Assign the map fragment variable on resume. This is necessary for orientation change events */
@Override
public void onResume ()
{
super.onResume();
// If the map fragment hasn't been set, but this is the current item, get the map fragment from the parent
final EventReportAlbumActivity activity = (EventReportAlbumActivity)getActivity();
if( m_mapFragment == null && this == activity.getCurrentItemFragment() )
{
setMapFragment( activity.getMapFragment() );
}
//
configureMap();
}
}
更新:上面的setMapFragment在有多个页面并且用户页面在旋转后通过相册时也会崩溃
经过一些测试,我也在相册有多个页面的情况下测试了这段代码。没有旋转,我可以毫无问题地滚动相册。但是,如果我旋转设备并尝试滚动相册,应用程序就会崩溃。同样,使用版本 19 的支持库,此代码按预期工作。这只是在升级到版本 22 后才开始发生。
在EventAlbumItemFragment中的setMapFragment()方法中:
fragmentManager.executePendingTransactions();
我得到以下堆栈跟踪:
java.lang.IllegalStateException: Could not execute method of the activity
at android.view.View$1.onClick(View.java:4020)
at android.view.View.performClick(View.java:4780)
at android.view.View$PerformClick.run(View.java:19866)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5254)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
Caused by: java.lang.reflect.InvocationTargetException
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at android.view.View$1.onClick(View.java:4015)
at android.view.View.performClick(View.java:4780)
at android.view.View$PerformClick.run(View.java:19866)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5254)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'java.lang.Object java.util.ArrayList.set(int, java.lang.Object)' on a null object reference
at android.support.v4.app.FragmentManagerImpl.makeInactive(FragmentManager.java:1192)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1099)
at android.support.v4.app.FragmentManagerImpl.removeFragment(FragmentManager.java:1235)
at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:710)
at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1501)
at android.support.v4.app.FragmentManagerImpl.executePendingTransactions(FragmentManager.java:490)
at crc.carsapp.fragments.MapAlbumItemFragment.setMapFragment(MapAlbumItemFragment.java:78)
at crc.carsapp.listeners.OnMapAlbumScrollListener.onPageSelected(OnMapAlbumScrollListener.java:46)
at crc.carsapp.listeners.OnEventAlbumViewScrollListener.onPageSelected(OnEventAlbumViewScrollListener.java:34)
at android.support.v4.view.ViewPager.dispatchOnPageSelected(ViewPager.java:1786)
at android.support.v4.view.ViewPager.scrollToItem(ViewPager.java:568)
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:552)
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:513)
at android.support.v4.view.ViewPager.setCurrentItem(ViewPager.java:505)
at crc.carsapp.activities.AlbumActivity.scrollToPrevious(AlbumActivity.java:133)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at android.view.View$1.onClick(View.java:4015)
at android.view.View.performClick(View.java:4780)
at android.view.View$PerformClick.run(View.java:19866)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5254)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
【问题讨论】:
-
您的 Google Play 服务是最新的吗?
-
是的,是7.5.0版
标签: android google-maps screen-rotation supportmapfragment