Como depurar a linha Android java.lang.IndexOutOfBoundsException HeaderViewListAdapter.java

Eu tenho um aplicativo que é ao vivo no playstore android, onde recentemente comecei a ver relatórios de falhas da Crashlytics com o seguinte rastreamento:

Fatal Exception: java.lang.IndexOutOfBoundsException: Invalid index 0, size is 0 at java.util.ArrayList.throwIndexOutOfBoundsException(ArrayList.java:251) at java.util.ArrayList.get(ArrayList.java:304) at android.widget.HeaderViewListAdapter.isEnabled(HeaderViewListAdapter.java:164) at android.widget.ListView.dispatchDraw(ListView.java:3259) at android.view.View.draw(View.java:14183) at android.widget.AbsListView.draw(AbsListView.java:4960) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.draw(View.java:14183) at android.widget.FrameLayout.draw(FrameLayout.java:467) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.getDisplayList(View.java:13116) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.support.v4.widget.DrawerLayout.drawChild(DrawerLayout.java:1229) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.draw(View.java:14183) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.draw(View.java:14183) at android.widget.FrameLayout.draw(FrameLayout.java:467) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.draw(View.java:14183) at android.support.v7.internal.widget.ActionBarOverlayLayout.draw(ActionBarOverlayLayout.java:500) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.getDisplayList(View.java:13116) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.getDisplayList(View.java:13116) at android.view.View.getDisplayList(View.java:13162) at android.view.View.draw(View.java:13896) at android.view.ViewGroup.drawChild(ViewGroup.java:3024) at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2893) at android.view.View.draw(View.java:14183) at android.widget.FrameLayout.draw(FrameLayout.java:467) at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:2449) at android.view.View.getDisplayList(View.java:13118) at android.view.View.getDisplayList(View.java:13162) at android.view.HardwareRenderer$GlRenderer.draw(HardwareRenderer.java:1198) at android.view.ViewRootImpl.draw(ViewRootImpl.java:2431) at android.view.ViewRootImpl.performDraw(ViewRootImpl.java:2303) at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2109) at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1179) at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:4859) at android.view.Choreographer$CallbackRecord.run(Choreographer.java:749) at android.view.Choreographer.doCallbacks(Choreographer.java:562) at android.view.Choreographer.doFrame(Choreographer.java:532) at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:735) at android.os.Handler.handleCallback(Handler.java:725) at android.os.Handler.dispatchMessage(Handler.java:92) at android.os.Looper.loop(Looper.java:137) at android.app.ActivityThread.main(ActivityThread.java:5328) at java.lang.reflect.Method.invokeNative(Method.java) at java.lang.reflect.Method.invoke(Method.java:511) at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1102) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:869) at dalvik.system.NativeStart.main(NativeStart.java) 

Meu aplicativo tem muitas canvass e visualizações, e deste traço não há quadro que é do meu aplicativo 🙁 Como faço para descobrir onde começar a olhar? Qualquer pointers sobre isso muito apreciado.

Substitua getHeadersCount() e sempre retorne 0 quando o tamanho dos dados for 0.

 @Override public int getHeadersCount(){ return getCount() == 0 ? 0 : super.getHeaderCount(); } 

Isso acontece quando você está tendo uma lista de dados não nulos e tenta acessar o elemento 0.

Você também pode precisar fazer o mesmo com o método isEnabled()

 @Override public boolean isEnabled(int position){ return position > 0 || getHeadersCount() > 0 ? super.isEnabled(position) : false; } 

Eu estava recebendo a mesma exceção ao usar o adaptador personalizado com ListView. E exceção foi lançada de classs de biblioteca padrão do Android, nem mesmo levando a qualquer linha do meu código. Eu também estava adicionando Header, então meu adaptador foi envolvido implicitamente pelo HeaderViewListAdapter. No meu caso, o problema aparece quando estou excluindo dados do adaptador.

Eu estava pensando que o problema é porque ListView ou adaptador não pode funcionar bem com o header, mas para valer a razão era outra.

Minha solução é certificar-se de que os dados do adaptador nunca sejam alterados de outros threads, e notifyDataSetChanged() é chamado de thread de interface do usuário.

Então, depois de consertar tudo funciona e meu código se parece com:

 // Loader's callbacks @Override public Loader<...> onCreateLoader(int id, Bundle args) { return new Loader... } @Override public void onLoadFinished(Loader<...> loader, Data data) { ... // adapter's list of items listItems.clear(); listItems.addAll(data); adapter.notifyDataSetChanged(); ... } @Override public void onLoaderReset(Loader>> loader) { ... } // custom loader private static class ContactAsyncLoader extends AsyncTaskLoader<...> { @Override public List<..> loadInBackground() { Data data = new .. ... return data; } }