使第一个ListView项目以不同的布局显示

时间:2015-07-22 08:31:09

标签: android listview getview

我正在尝试制作一个 ListView,其中第一个项目以不同的布局显示,而其他项目则采用通用布局。两种布局都具有相同的元素,这些元素成对具有相同的名称。当我这样做时:

public View getView(int position, View convertView, ViewGroup parent) {        
    if(convertView == null) {
        if(position == 0){
            convertView = inflater.inflate(R.layout.article_list_top_item, parent, false);
            Log.d("ALA", "pos = " + position + ", inflated top");
        }
        else {
            convertView = inflater.inflate(R.layout.article_list_item, parent, false);
            Log.d("ALA", "pos = " + position + ", inflated normal");
        }
    }
    // setText, setBitmap etc here
    return convertView;
}

它不起作用。 从我可以说的日志中,inflater.inflate被触发6次,inflated top 1次,inflated normal 5次。

显示的是,文章[0]在布局article_list_top_item中,文章[1]〜文章[5]在article_list_item中。

直到这里它没关系,但重复的模式,这意味着文章[6],文章[12],[18],......都在布局article_list_top_item,这不是我想要的。

我该怎么办才能使第一篇文章出现在 article_list_top_item ??

P.S。我尝试重新命名article_list_top_item.xml中的元素,如果分支setText setImage进程,则无效。

我尝试在else return convertView;行之前添加//setText,它变得一团糟。

我想过为第一个项目制作专用的布局元素,但这不是我想要的,因为整个列表位于SwipeRefreshLayout

之下

请帮忙。

3 个答案:

答案 0 :(得分:5)

您应该使用不同的itew视图类型。你的适配器重用已经膨胀的视图,所以如果你想要不同的类型,你应该告诉他:

为每种视图类型

创建一个持有者
private class FirstHolder {
    //add a field for each subview in view type 1
}
private class SecondHolder {
    //add a field for each subview in view type 2
}

覆盖getViewTypeCount()方法

@Override
public int getViewTypeCount() {
    return 2;
}

覆盖getItemViewType(int position)方法。此方法告知每个位置使用哪种类型的视图。

@Override
public int getItemViewType(int position) {
    if (position == 0) {
        return 0;
    } else {
        return 1;
    }
}

构建并设置视图:

public View getView(int position, View convertView, ViewGroup parent) {
    if (position == 0) {
        FirstHolder holder;
        if (convertView == null) {
            holder = new FirstHolder();
            convertView = inflater.inflate(R.layout.whatever_firstlayout, parent, false);
            //for each field of holder find the subview
            convertView.setTag(holder);
        } else {
            holder = (FirstHolder) convertView.getTag();
        }
        //set the data in subview with holder fields
    } else {
        SecondHolder holder;
        if (convertView == null) {
            holder = new SecondHolder();
            convertView = inflater.inflate(R.layout.whatever_secondlayout, parent, false);
            //for each field of holder find the subview
            convertView.setTag(holder);
        } else {
            holder = (SecondHolder) convertView.getTag();
        }
        //set the data in subview with holder fields
    }
    return convertView;
}

答案 1 :(得分:1)

您可以使用addHeaderView。例如;

LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View head = inflater.inflate(R.layout.headerlayout, listview, false);
listview.addHeaderView(head);

答案 2 :(得分:0)

我发现了问题。正如@sonic说的那样,我想对同一个ListView使用不同类型的视图,所以我必须通过覆盖getViewTypeCount()getItemViewType(int position)来告诉他。工作代码:

@Override
public int getViewTypeCount(){
    return 2;
}

@Override
public int getItemViewType(int pos){
    return pos > 0 ? 1 : 0;
}

public View getView(int position, View listItem, ViewGroup parent) {
    if(listItem == null) 
        listItem = inflater.inflate(getItemViewType(position) == 0 ?
                   R.layout.article_list_top_item : R.layout.article_list_item
                   , parent, false);
    // setText setBitmap etc. here
}

评论:即使我不在任何地方使用getViewTypeCount()getItemViewType(int position),它仍然有效,但重要的是它们很重要。疑 getItemViewType(int position)会做任何事情,我试着不要覆盖它 - 结果是,我必须重写两者才能使它工作。

我想知道怎么做。也许每当视图使用之前从未使用过的布局进行充气时,系统会为该布局分配一个从0到getViewTypeCount() - 1的数字?否则,它在哪里使用(在父类的代码中)使它重要?

我会留在这里稍后再思考。