我正在尝试制作一个 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
请帮忙。
答案 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
的数字?否则,它在哪里使用(在父类的代码中)使它重要?
我会留在这里稍后再思考。