java - studio - ArrayIndexOutOfBoundsException cuando se utiliza ListAdapter
llenar un listview android (4)
Estoy usando un ListAdapter
personalizado para un ListView
. No cambio nada dinámicamente (ver conteo, ver tipos, habilitar / deshabilitar, leo otras publicaciones pero cambiaron algo del código). Tengo que hacer lo siguiente para activar la excepción. Desplázate hasta el final de la lista y luego desplázate hacia arriba nuevamente. Al comienzo de la animación de desplazamiento obtengo esta excepción.
04-14 09:41:43.907: ERROR/AndroidRuntime(400): FATAL EXCEPTION: main
04-14 09:41:43.907: ERROR/AndroidRuntime(400): java.lang.ArrayIndexOutOfBoundsException
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:4540)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.widget.AbsListView.trackMotionScroll(AbsListView.java:3370)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.widget.AbsListView.onTouchEvent(AbsListView.java:2233)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.widget.ListView.onTouchEvent(ListView.java:3446)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.View.dispatchTouchEvent(View.java:3885)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:903)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:942)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:942)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:942)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1691)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1125)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.app.Activity.dispatchTouchEvent(Activity.java:2096)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1675)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewRoot.deliverPointerEvent(ViewRoot.java:2194)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.view.ViewRoot.handleMessage(ViewRoot.java:1878)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.os.Handler.dispatchMessage(Handler.java:99)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.os.Looper.loop(Looper.java:123)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at android.app.ActivityThread.main(ActivityThread.java:3683)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at java.lang.reflect.Method.invokeNative(Native Method)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at java.lang.reflect.Method.invoke(Method.java:507)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
04-14 09:41:43.907: ERROR/AndroidRuntime(400): at dalvik.system.NativeStart.main(Native Method)
Abajo se encuentra la implementación completa.
public class CompanyDetailsAdapter implements ListAdapter {
private Context context;
private LayoutInflater inflater;
private static final class ViewType {
private static final int HEADER = 1;
private static final int TEXT = 2;
private static final int ADDRESS = 3;
private static final int REVIEWS = 4;
private static final int PRODUCTS = 5;
}
//all the cells in the order they appear
private static final class ViewPositions {
private static final int CompanyNameHeader = 0;
private static final int CompanyName = 1;
private static final int ContactHeader = 2;
private static final int Phone = 3;
private static final int Website = 4;
private static final int AddressHeader = 5;
private static final int Address = 6;
private static final int DescriptionHeader = 7;
private static final int Description = 8;
private static final int ReviewsHeader = 9;
private static final int Reviews = 10;
private static final int ProductsHeader = 11;
private static final int Products = 12;
}
//list of cells by types
private static final int[] viewTypes = {
ViewType.HEADER, //0
ViewType.TEXT, //1
ViewType.HEADER, //2
ViewType.TEXT, //3
ViewType.TEXT, //4
ViewType.HEADER, //5
ViewType.ADDRESS, //6
ViewType.HEADER, //7
ViewType.TEXT, //8
ViewType.HEADER, //9
ViewType.REVIEWS, //10
ViewType.HEADER, //11
ViewType.PRODUCTS //12
};
//headers
private static final int[] headerPositions = {
ViewPositions.CompanyNameHeader,
ViewPositions.ContactHeader,
ViewPositions.AddressHeader,
ViewPositions.DescriptionHeader,
ViewPositions.ReviewsHeader,
ViewPositions.ProductsHeader
};
//header resources for position where needed
private static final int[] headerResources = {
R.string.companyName,
0,
R.string.contact,
0,
0,
R.string.address,
0,
R.string.description,
0,
R.string.reviews,
0,
R.string.products,
0
};
//regular texts
private static final int[] textPositions = {
ViewPositions.CompanyName,
ViewPositions.Phone,
ViewPositions.Website,
ViewPositions.Description,
};
public CompanyDetailsAdapter(Context context) {
this.context = context;
this.inflater = LayoutInflater.from(this.context);
}
public int getCount() {
return viewTypes.length;
}
public Object getItem(int position) {
return null;
}
public long getItemId(int position) {
return position+1;
}
public int getItemViewType(int position) {
return viewTypes[position];
}
public View getView(int position, View convertView, ViewGroup parent) {
if (arrayContains(headerPositions, position)) {
LinearLayout layout = null;
TextView txtHeader = null;
if (convertView == null) {
layout = (LinearLayout)inflater.inflate(R.layout.header_listview,parent, false);
} else {
layout = (LinearLayout)convertView;
}
txtHeader = (TextView)layout.findViewById(R.id.txtHeader);
String value = context.getText(headerResources[position]).toString();
txtHeader.setText(value);
return layout;
} else if (arrayContains(textPositions, position)) {
TextView txtText = null;
LinearLayout layout = null;
Company c = Company.getSelectedCompany();
if (convertView == null) {
layout = (LinearLayout)inflater.inflate(R.layout.default_text,parent, false);
} else {
layout = (LinearLayout)convertView;
}
txtText = (TextView)layout.findViewById(R.id.txtNormalText);
switch (position) {
case ViewPositions.CompanyName:
txtText.setText(c.getName());
break;
case ViewPositions.Phone:
txtText.setText(c.getPhone());
break;
case ViewPositions.Website:
txtText.setText(c.getWebsiteString());
break;
case ViewPositions.Description:
txtText.setText(c.getDescription());
break;
}
return layout;
} else {
View v = null;
//LinearLayout layout = null;
Company company = Company.getSelectedCompany();
switch (position) {
case ViewPositions.Address:
if (convertView == null)
v = (LinearLayout)inflater.inflate(R.layout.adress, parent,false);
else
v = (LinearLayout)convertView;
TextView txtAddress = (TextView)v.findViewById(R.id.txtAddress);
txtAddress.setText(String.format("%s %s %s", company.getCity(),
company.getStreet(), company.getPostalCode()));
break;
case ViewPositions.Products:
if (convertView == null)
v = (LinearLayout)inflater.inflate(R.layout.products, parent,false);
else
v = (LinearLayout)convertView;
v = (LinearLayout)inflater.inflate(R.layout.products, parent,false);
TextView txtNumProducts = (TextView)v.findViewById(R.id.txtNumProducts);
txtNumProducts.setText(String.valueOf(company.getNrProducts()));
break;
case ViewPositions.Reviews:
if (convertView == null)
v = (LinearLayout)inflater.inflate(R.layout.reviews, parent,false);
else
v = (LinearLayout)convertView;
v = (LinearLayout)inflater.inflate(R.layout.reviews, parent,false);
TextView txtNumReviews = (TextView)v.findViewById(R.id.txtNumReviews);
txtNumReviews.setText(String.valueOf(company.getNrReviews()));
RatingBar rating = (RatingBar)v.findViewById(R.id.ratAvg);
rating.setRating(company.getAvgRating());
break;
}
return v;
}
}
public int getViewTypeCount() {
return 5;
}
public boolean hasStableIds() {
return true;
}
public boolean isEmpty() {
return false;
}
public void registerDataSetObserver(DataSetObserver arg0) {
}
public void unregisterDataSetObserver(DataSetObserver arg0) {
}
public boolean areAllItemsEnabled() {
return false;
}
public boolean isEnabled(int position) {
if (arrayContains(headerPositions, position)) return false;
return true;
}
private Boolean arrayContains(int[] array, int element) {
for (int i=0; i<array.length; i++) {
if (array[i] == element) return true;
}
return false;
}
}
Anexé las fuentes a la biblioteca de Android 2.3.3 desde here . No encontré nada más que el rastro de la pila no me mostró. Como todos saben, el ListView
recicla vistas. Hay una colección de algún tipo que contiene estas vistas recicladas. Cuando desaparece una vista, se agrega allí (creo). Esto sucede en at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:4540)
En algún lugar de aquí me da una excepción de que algo está fuera de límites. No tengo la menor idea de qué. Pero no tengo la paciencia para depurar otras 1000 líneas de código, así que creo que usaré ScrollView por ahora.
Parece que es esta línea: (AbsListView.class: 4540 mScrapViews[viewType].add(scrap);
mScrapViews es de tipo private ArrayList<View>[] mScrapViews;
una matriz de listas de tipo View
. Puedo intentarlo para ver scrap
variables viewType
y scrap
pero dice errors_during_evaluation
. Probablemente sea demasiado tonto para evaluar.
Cambiando esto:
public int getViewTypeCount()
{
return 2;
}
a esto me ayudó:
public int getViewTypeCount()
{
return 3;
}
Explicación:
Inicialmente tenía 2 tipos de vista, pero más tarde en la implementación, agregué un tercero. Empecé a recibir esa excepción porque olvidé actualizar el método getViewTypeCount () para devolver 3 en lugar de solo 2.
Espero que esto ayude a alguien :)
Encontrado el problema Los tipos de vista deben comenzar en 0, el mío comienza en 1.
Android está usando los tipos de vista que proporcioné para acceder a una matriz por índice en alguna parte. SomeArraySomewhere[myViewType]
El tamaño de esta matriz probablemente sea igual al número de tipos de vista. Entonces esto significa que debo comenzar mis tipos de vista en 0.
Es posible que la position
sea:
- a:
null
- b:
higher then the amount of headerResources
?
En esta línea (getView):
String value = context.getText(headerResources[position]).toString();
Primero aprende a depurar: aquí hay un buen video tutorial
La razón es muy probable porque el método getItemViewType está devolviendo los valores incorrectos. Cada fila en la vista de lista es una Vista. Mientras scrooling getItemViewType alcanza más que el conteo de View.
¿Qué hacer? ¿Cómo evitar el problema?
Primero determine la vista (fila) en su vista de lista que se muestra primero. Mientras se desplaza utiliza la ecuación de modulador
@Override
public int getItemViewType(int position) {
return choseType(position);//function to use modular equation
}
@Override
public int getViewTypeCount() {
return 10;
}
En este ejemplo, hay diez vistas (10 filas).
private int choseType(int position){
if(position%10==0)
return 0;
else if(position%10==1)
return 1;
else if(position%10==2)
return 2;
else if(position%10==3)
return 3;
else if(position%10==4)
return 4;
else if(position%10==5)
return 5;
else if(position%10==6)
return 6;
else if(position%10==7)
return 7;
else if(position%10==8)
return 8;
else
return 9;
}
Importante
Algunos usuarios mencionaron en otra pregunta sobre ese método
public int getViewTypeCount () y public int getItemViewType (int position) fix como Tooglebutton habilita automáticamente state check true en scrooling ... eso está muy mal . Si no quieres enbale automático en scrool solo hazlo
toogleButton.setChecked(false);
en el método de anulación getView.