使第一个 ListView 项目出现在不同的布局中

Make first ListView item appear in a different layout

我正在尝试创建一个 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中。

到这里没问题,但是模式重复了,也就是说article[6],article[12],[18],...都在布局article_list_top_item中,这不是我的想。

我怎样才能使第一篇文章出现在 article_list_top_item??

P.S。我尝试重命名 article_list_top_item.xml 中的元素并分支 setText setImage 过程,但没有帮助。

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

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

请帮忙。

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

为您的每种视图类型创建一个支架

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;
}

您可以使用addHeaderView。例如;

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

我发现了问题。正如@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 的数字?否则,它在哪里使用(在父 类' 代码中)使其变得重要?

留在这里以后再想。