I\'ve been working with Google Maps API v2 (version 3.2.25 (761454-30)) on my Android app, and sometimes it works perfectly though sometimes (which is getting each time more
Well, I managed to "fix" this issue. It isn't actually a fix but at least I'm avoiding a crash on my app when that exception is thrown. What I did was to override the default uncaught exception handler and suppress crashing the app when the uncaught exception is this one. The crash was because of a NullPointerException and it always happened on a GLThread (must be from the drawing code from Google Maps). So I check if all the prerequisites apply, and if so, I send a broadcast with a custom action so I can listen for that broadcast on the fragment on which I use a Google Map, replacing my SupportMapFragment when I get one. Since we got an uncaught exception, the GLThread is interrupted and the drawing to the Google Maps Fragment is suspended, so we do need to replace the Google Maps Fragment for the map to work again.
Here is the code I had to add:
On my Application's onCreate:
final UncaughtExceptionHandler defaultHandler = Thread.getDefaultUncaughtExceptionHandler();
Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionHandler() {
@Override
public void uncaughtException(Thread thread, Throwable ex) {
if (ex instanceof NullPointerException && thread.getName().startsWith("GLThread")) {
for (StackTraceElement stackTraceElement : ex.getStackTrace()) {
if (stackTraceElement.getClassName().contains("maps")) {
sendBroadcast(new Intent(ACTION_MAPS_NPE));
return;
}
}
}
defaultHandler.uncaughtException(thread, ex);
}
});
And then on my Fragment which uses a Google Maps Fragment, I register this broadcast receiver right before adding the SupportMapFragment to the layout on onCreateView (and unregister it on onDestroyView):
private class GoogleMapsNPEReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Fragment mapFragment = getChildFragmentManager().findFragmentById(R.id.map_fragment_container);
if (mapFragment != null) {
getChildFragmentManager().beginTransaction()
.replace(R.id.map_fragment_container, SupportMapFragment.newInstance())
.commit();
mMapInitialized = false;
// We won't have onResume to initialize our map anymore. Try to initialize it after 100ms.
final Handler handler = new Handler();
handler.postDelayed(new Runnable() {
@Override
public void run() {
if (getGoogleMap() == null) {
handler.postDelayed(this, 100);
}
}
}, 100);
}
}
}
Hope this helps anyone with that same issue. If anyone has an actual fix for it please let me know.