事件问题:
---------------------------------------------------------
1、 item内如果有button等控件时,在监听listview的onitemclick事件时,焦点会被item内的button、 imagebutton等控件抢走,
从而导致在listview设置了onitemclick事件后不会被触发。解决方法是在初始化item的时候屏蔽掉 其内部button等控件的焦点获取,
具体方法可以在自定义item的根控件中调用:
setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS);
这样就能阻塞子控件抢夺焦点,listview的onitemclick就能被正确触发,同时对item内部的button等控件也没有影响,
他们在被点击时照样可以触发自身的点击事件。
2、 当listview需要添加headerview时,可以通过调用listview的addHeaderView(headView, null, false) 方法,
该方法还有一个重载方法 addHeaderView(headView);这两个方法的区别是前一个方法可以控制header是否可以被 selected,
如果不想被selected则将第三个参数设置成false;
3、 接着上面说的添加header,添加header时调用的addHeaderView方法必须放在listview.setadapter前面,
意思很明 确就是如果想给listview添加头部 则必须在给其绑定adapter前添加,否则会报错。
原因是当我们在调用setAdapter方法时会 android会判断当前listview是否已经添加header,
如果已经添加则会生成一个新的tempadapter,这个新的 tempadapter包含我们设置的adapter所有内容以及listview的header和footer。
所以当我们在给listview添加了 header后在程序中调用listview.getadapter时返回的是tempadapter而不是我们通过setadapter传进去的 adapter。
如果没有设置adapter则tempadapter与我们自己的adapter是一样的。 listview.getadapter().getcount()方法返回值会比我们预期的要大,
原因是添加了header。
4、接着上面的tempadapter说,我们自定义adapter里面的getitem方法里面返回的position是不包括header的,
是我们自定义adapter中数据position编号从0开始,也就是说与我们传进去的list的位置是一样的。
@Override
public View getView(int position, View convertView, ViewGroup parent) {
// TODO Auto-generated method stub
Log.i("adapter", "position:"+position); //这个position就是我们数据的真实位置
}
而listview的onitemclick方法中:
public void onItemSelected(AdapterView<?> parent, View view, int position, long id)
position是当前click的位置,这个位置是指在HeaderViewListAdapter中的位置,从0开始如果listview中添加了header则0代表header。
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- // TODO Auto-generated method stub
- Log.i("adapter", "position:"+position); //这个position就是我们数据的真实位置
- }
而listview的onitemclick方法中:
- public void onItemClick(AdapterView
headView和footerView都可以响应onItemClick方法,headView的position为0, footerView的position最大。
不过可以给headView和footerView设置OnClickListener来覆盖OnItemClick,
这样,你点击headview或者footerView将触发OnClickListener而不是onItemClick().
布局问题:
---------------------------------------------------------
在做ListView上拉加载中遇到的问题,不清楚什么原理,具体也没仔细研究,Mark一下。
布局界面如下:
foot1.xml
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:background="#f1f1f1" > <TextView android:layout_width="wrap_content" android:layout_height="50dip" android:gravity="center" android:text="查看更多" android:textSize="18sp" /> </RelativeLayout>
foot2.xml
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="50dip" android:gravity="center" android:background="#f1f1f1" > <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:gravity="center" android:text="查看更多" android:textSize="18sp" /> </RelativeLayout>
看似效果一样的两个布局界面,在ListView的FooterView中显示却是两个效果。
footer1.xml中设置了TextView的高度为50dip,而根布局则设置为包裹内容,这样在实际显示效果中ListView的Footer确实高度是50dip了,
但是footer2.xml中在根布局中设置了高度为50dip,在实际加载到ListView的FooterView中后却没有效果,
他的高度变为了wrap_content,而TextView的高度也设置为wrap_content,所以实际显示出来的高度就是TextView内容的高度,也就是字体的高度。
********************************************************************
如果需要动态的显示和隐藏ListView的header和footer的话,以前以为直接setVisibility中的View.GONE就可以。
但是在实际使用中发现并不是这样的。直接设置GONE后,虽然元素是隐藏了,
但是还是占用着那个区域。感觉就像View.INVISIBILE一样的效果了。后来在网上查找后发现。
只要给footer再加上一个layout或者是父级的view就可以了。
例如:(示例代码,header同理)
1
2
3
|
private
ListView listView =
new
ListView(context);
private
TextView footer =
new
TextView(context);
|
然后直接
1
2
3
|
listView.addFooterView(footer);
footer.setVisibility(View.GONE);
|
这样的话,是不行的。需要
1
2
3
4
5
6
7
|
LinearLayout footerParent =
new
LinearLayout(context);
footerParent.addView(footer);
listView.addFooterView(footerParent);
footer.setVisibility(View.GONE);
|
这样就可以达到真正想要的效果了。
在知道这个效果之前,都是用remove和add方法,比较郁闷。这样的方式感觉是很迂回的,而且还得加多一个boolean的变量来判断。
关于FooterView的添加和删除
* 3.1每次总是先remove掉FooterView
* 3.2若有需求再add上FooterView
我们通常在加载数据时,为了省流量不会一次性把数据全部下完,一般是分段下载。
分段下载一般会在listview最后面放一个进度条表示正在加载数据,当数据加载完时,我们又要清除它。这时候就要注意了。
mLoadingLayout = (FrameLayout) View.inflate(this, R.layout.load, null);
listView.addFooterView(mLoadingLayout);
listView.requestFocus();
这是listview尾部添加一个进度条。
listView.removeFooterView(mLoadingLayout);
这是移除尾部的进度条。
有时候在移除时回报空指针,但listview不为null ,mLoadingLayout也不为null,但还是报空指针,原因是因为listview要分为三部分。
一是头部,二是中间部,三是尾部。在设置了头部或尾部时,必须要有中间部才能真正意义上的生效。没生效就去移除就会报空指针错误。
所以在
listView.removeFooterView(mLoadingLayout);时
必须先调用 listView.setAdapter(adapter);(设置中间部)
adapter可以数据可以为0但不可为null