SectionedAdapter ArrayIndexOutOfBoundsException в AbsListView $ RecycleBin.addScrapView () Проблема? - PullRequest
2 голосов
/ 20 мая 2011

У меня есть секционный адаптер, который я использую, однако я пытаюсь изменить его, чтобы секции не отображались, если они пусты, используя if (section.adapter.getCount ()> 0). У меня есть следующий класс:

abstract public class SectionedAdapter extends BaseAdapter {

abstract protected View getHeaderView(String caption, int index, View convertView, ViewGroup parent);

private List<Section> sections = new ArrayList<Section>(); 
private static int TYPE_SECTION_HEADER = 0; 

public SectionedAdapter() {
    super(); 
}

public void addSection(String caption, Adapter adapter) { 
    sections.add(new Section(caption, adapter)); 
}

public void clear() {
    sections.clear();
    notifyDataSetInvalidated();
}

@Override
public Object getItem(int position) { 

    for (Section section : this.sections) {
        if (section.adapter.getCount() > 0) {

            if (position == 0) {
                return section;
            }

            int size = section.adapter.getCount() + 1;

            if (position < size) {
                return section.adapter.getItem(position - 1);
            }

            position -= size; 
        }
    } 

    return null; 
}

@Override
public int getCount() {
    int total = 0; 
    for (Section section : this.sections) {
        if (section.adapter.getCount() > 0) {
            // Add Header and Section
            total += section.adapter.getCount() + 1;
        }
    }
    return total;
} 

@Override
public int getViewTypeCount() {
    int total = 1; // Add Header
    for (Section section : this.sections) {
        if (section.adapter.getCount() > 0) {
            // Add Sections
            total += section.adapter.getViewTypeCount();
        }
    } 
    return total;
} 

@Override
public int getItemViewType(int position) { 
    int typeOffset = TYPE_SECTION_HEADER + 1;

    for (Section section : this.sections) {
        if (section.adapter.getCount() > 0) {

            if (position == 0) {
                return TYPE_SECTION_HEADER;
            }

            int size = section.adapter.getCount() + 1;

            if (position < size) {
                return (typeOffset + section.adapter.getItemViewType(position - 1));
            }

            position -= size; 
            typeOffset += section.adapter.getViewTypeCount();
        }
    } 

    return Adapter.IGNORE_ITEM_VIEW_TYPE;
}

@Override
public boolean areAllItemsEnabled() { 
    return false; 
}

@Override
public boolean isEnabled(int position) { 
    return (getItemViewType(position) != TYPE_SECTION_HEADER); 
}

@Override
public View getView(int position, View convertView, ViewGroup parent) { 
    int sectionIndex = 0;
    for (Section section : this.sections) {
        if (section.adapter.getCount() > 0) {

            if (position == 0) {
                return getHeaderView(section.caption, sectionIndex, null, parent);
            }

            int size = section.adapter.getCount() + 1;

            if (position < size) {
                return section.adapter.getView(position - 1, null, parent);
            } 

            position -= size;
            sectionIndex++;
        }
    }
    return null; 
} 

@Override
public long getItemId(int position) { 
    return position; 
}

class Section { 
    String caption; 
    Adapter adapter; 

    Section(String caption, Adapter adapter) { 
        this.caption = caption; 
        this.adapter = adapter; 
    } 
}

}

Это работает, но иногда происходит сбой с ошибкой ArrayIndexOutOfBoundsException в AbsListView $ RecycleBin.addScrapView ()

java.lang.ArrayIndexOutOfBoundsException
at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:4186)
at android.widget.ListView.layoutChildren(ListView.java:1572)
at android.widget.AbsListView.onLayout(AbsListView.java:1172)
at android.view.View.layout(View.java:7037)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1249)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1125)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1042)
at android.view.View.layout(View.java:7037)
at android.widget.FrameLayout.onLayout(FrameLayout.java:333)
at android.view.View.layout(View.java:7037)
at android.widget.FrameLayout.onLayout(FrameLayout.java:333)
at android.view.View.layout(View.java:7037)
at android.view.ViewRoot.performTraversals(ViewRoot.java:1054)
at android.view.ViewRoot.handleMessage(ViewRoot.java:1736)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:143)
at android.app.ActivityThread.main(ActivityThread.java:4701)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:521)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:868)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:626)
at dalvik.system.NativeStart.main(Native Method)

Я знаю, что эта ошибка, скорее всего, связана с getViewTypeCount () или getItemViewType (int position), но я не могу понять, где я иду неправильно (из-за других запросов StackOverflow). Исключение не возникает надежно, поэтому я не понимаю, почему это происходит. Я что-то здесь упускаю?

1 Ответ

0 голосов
/ 20 мая 2011

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

Итак, удалите условия, которые вы поставили в SectionedAdapter.getHeaderView здесь абстрактный метод.Но реализация написана в каком-то другом классе, и вы должны изменить его, как я.

SectionedAdapter adapter = new SectionedAdapter() {
        protected View getHeaderView(String caption, int index, View convertView, ViewGroup parent) {
            TextView result = (TextView)convertView;

            if (convertView == null) {
                result=(TextView)getLayoutInflater().inflate(R.layout.header, null);
            }
            if(caption != null) {
                result.setText(caption);
            } else {
                result.setVisibility(View.GONE);
            }

            return(result);
        }
    };

Если вы не хотите вставлять какой-либо раздел, тогда в addSelection() передайте метод null вместо имени раздела.

Пожалуйста, посмотрите на мой модифицированный код.попробуйте запустить класс SectionedDemo

// SectionedDemo class
//----------------------
import android.app.ListActivity;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

public class SectionedDemo extends ListActivity {
    private static String[] items={ "lorem", "ipsum", "dolor", "sit", "amet", "consectetuer", "adipiscing", 
                                    "elit", "morbi", "vel", "ligula", "vitae", "arcu", "aliquet", "mollis", 
                                    "etiam", "vel", "erat", "placerat", "ante", "porttitor", "sodales", 
                                    "pellentesque", "augue", "purus"};

    @Override
    public void onCreate(Bundle icicle) {
        super.onCreate(icicle);
        setContentView(R.layout.main1);

        adapter.addSection(null, new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, items));

        List<String> list = Arrays.asList(items);
        Collections.shuffle(list);
        adapter.addSection(null, new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, list));

        list = Arrays.asList(items);
        Collections.shuffle(list);
        adapter.addSection(null, new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, list));

        setListAdapter(adapter);
    }

    SectionedAdapter adapter = new SectionedAdapter() {
        protected View getHeaderView(String caption, int index, View convertView, ViewGroup parent) {
            TextView result=(TextView)convertView;

            if (convertView==null) { 
                result=(TextView)getLayoutInflater() .inflate(R.layout.header, null);
            }
            if(caption != null) {
                result.setText(caption);
            } else {
                result.setVisibility(View.GONE);
            }

            return(result);
        }
    };
}

И ниже приведен класс SectionedAdapter

//SectionedAdapter class
//-------------------------
import android.view.View;
import android.view.ViewGroup;
import android.widget.Adapter;
import android.widget.BaseAdapter;
import java.util.ArrayList;
import java.util.List;

abstract public class SectionedAdapter extends BaseAdapter {
    abstract protected View getHeaderView(String caption, int index, View convertView, ViewGroup parent);

    private List<Section> sections=new ArrayList<Section>();
    private static int TYPE_SECTION_HEADER=0;

    public SectionedAdapter() {
        super();
    }

    public void addSection(String caption, Adapter adapter) {
        sections.add(new Section(caption, adapter));
    }

    public Object getItem(int position) {
        for (Section section : this.sections) {
            if (position==0) {
                return(section);
            }

            int size=section.adapter.getCount()+1;

            if (position<size) {
                return(section.adapter.getItem(position-1));
            }

            position-=size;
        }

        return(null);
    }

    public int getCount() {
        int total = 0;

        for (Section section : this.sections) {
            total += section.adapter.getCount()+1; // add one for header
        }

        return(total);
    }

    public int getViewTypeCount() {
        int total = 1;  // one for the header, plus those from sections

        for (Section section : this.sections) {
            total += section.adapter.getViewTypeCount();
        }

        return(total);
    }

    public int getItemViewType(int position) {
        int typeOffset=TYPE_SECTION_HEADER+1;   // start counting from here

        for (Section section : this.sections) {
            if (position==0) {
                return(TYPE_SECTION_HEADER);
            }

            int size=section.adapter.getCount()+1;

            if (position<size) {
                return(typeOffset+section.adapter.getItemViewType(position-1));
            }

            position-=size;
            typeOffset+=section.adapter.getViewTypeCount();
        }

        return(-1);
    }

    public boolean areAllItemsSelectable() {
        return(false);
    }

    public boolean isEnabled(int position) {
        return(getItemViewType(position)!=TYPE_SECTION_HEADER);
    }

    public View getView(int position, View convertView, ViewGroup parent) {
        int sectionIndex=0;

        for (Section section : this.sections) {

            if (position==0) {
                return(getHeaderView(section.caption, sectionIndex, convertView, parent));
            }

            int size=section.adapter.getCount()+1;

            if (position<size) {
                return(section.adapter.getView(position-1, convertView, parent));
            }

            position-=size;
            sectionIndex++;
        }

        return(null);
    }

    public long getItemId(int position) {
        return(position);
    }

    class Section {
        String caption;
        Adapter adapter;

        Section(String caption, Adapter adapter) {
            this.caption=caption;
            this.adapter=adapter;
        }
    }
}
...