Android ExpandableListView长按事件的完美解决办法
程序员文章站
2023-12-01 20:01:40
关于expandablelistview长按事件处理,网上很多都是使用将上下文菜单注册到expandablelistview上实现长按事件。这样做弊端显而易见,不够灵活,不...
关于expandablelistview长按事件处理,网上很多都是使用将上下文菜单注册到expandablelistview上实现长按事件。
这样做弊端显而易见,不够灵活,不能分别对父项、子项、父项之间、子项之间弹出内容做区分。
下面来说我的解决方法,方法有点投机取巧。首先说明一点,使用我这种方法必须使用自定义的baseexpandablelistadapter,至于为什么,具体后面讲到。
expandablelistview本身有继承自adapterview的setonitemlongclicklistener(adapterview.onitemlongclicklistener listener)方法。
实现监听器:
/**
* 长按邮箱快捷选项
* @author king
*/
private class quickwaylistener implements onitemlongclicklistener{
@override
public boolean onitemlongclick(adapterview<?> arg0, view view,
int pos, long id) {
//pos不可用说明见下文
return false;
}
}
如果这个方法是用在listview长按事件中刚刚好,但在expandablelistview中,第三个参数pos不能区分开点击的是父项还是子项,以及哪个父项或子项。
在expandablelistview响应的onitemlongckick方法中,pos参数值为:从上到下,父项+展现的子项到点击位置的数目(注意:是展现的,隐藏的子项不包括,从0开始)。
例如:
父项1(隐藏3个子项)
父项2
|—子项2-0
|—子项2-1
|—子项2-2
长按子项2-1时,pos值为3。显然根据pos值是无法确定点击的是哪个子项或父项的。
因此依赖pos是很难处理点击位置的。
如果可以直接在onitemlongclick方法中获取grouppos,及childpos该多好呢?
于是看到了onitemlongclick方法第二个参数:view。这里的view是你按中的位置对应的view。view有个方法gettag(int key)。如果在创建此view的时候就把grouppos,childpos通过settag(int key, object value)设置进去,在响应onitemlongclick不就可以直接拿出来用了么。
现在就要讲到必须使用自定义的baseexpandablelistadapter的理由了。
要把grouppos,childpos通过settag的方式绑定到view中,就必须操作该view的创建过程。要控制这个过程就必须要在自定义baseexpandablelistadapter中重写getgroupview及getchildview方法进行操作。如下:
public class accountlistadapter extends baseexpandablelistadapter {
...省略其他方法
@override
public view getchildview(int groupposition, int childposition,
boolean islastchild, view convertview, viewgroup parent) {
//我这里仅通过自己写的mkchildview()方法创建textview来显示文字,更复杂的可以通过layoutinflater来填充一个view
textview childtv = mkchildview();
// 标记位置
// 必须使用资源id当key(不是资源id会出现运行时异常),android本意应该是想用tag来保存资源id对应组件。
// 将groupposition,childposition通过settag保存,在onitemlongclick方法中就可以通过view参数直接拿到了!
childtv.settag(r.id.xxx01, groupposition);
childtv.settag(r.id.xxx02, childposition);
return childtv;
}
@override
public view getgroupview(int groupposition, boolean isexpanded,
view convertview, viewgroup parent) {
textview grouptv = mkgroupview();
// 设置同getchildview一样
grouptv.settag(r.id.xxx01, groupposition);
grouptv.settag(r.id.xxx02, -1); //设置-1表示长按时点击的是父项,到时好判断。
grouptv.settext(groups[groupposition]);
return grouptv;
}
}
完成了这一步,我们只需要在expandablelistview响应的onitemlongclick方法时通过view.gettag(r.id.xxx01),view.gettag(r.id.xxx02)即可拿到grouppos,childpos.
如下:
/**
* 长按邮箱快捷选项
* @author king
*/
private class quickwaylistener implements onitemlongclicklistener{
@override
public boolean onitemlongclick(adapterview<?> arg0, view view,
int pos, long id) {
int grouppos = (integer)view.gettag(r.id.xxx01); //参数值是在settag时使用的对应资源id号
int childpos = (integer)view.gettag(r.id.xxx02);
if(childpos == -1){//长按的是父项
//根据grouppos判断你长按的是哪个父项,做相应处理(弹框等)
} else {
//根据grouppos及childpos判断你长按的是哪个父项下的哪个子项,然后做相应处理。
}
return false;
}
}
到这就写完了,貌似比较啰嗦。重写baseexpandablelistadapter写的比较简洁,没看明白的朋友可以先到网上查下怎么自定义baseexpandablelistadapter,和自定义baseadapter其实是一样的。
这样做弊端显而易见,不够灵活,不能分别对父项、子项、父项之间、子项之间弹出内容做区分。
下面来说我的解决方法,方法有点投机取巧。首先说明一点,使用我这种方法必须使用自定义的baseexpandablelistadapter,至于为什么,具体后面讲到。
expandablelistview本身有继承自adapterview的setonitemlongclicklistener(adapterview.onitemlongclicklistener listener)方法。
实现监听器:
复制代码 代码如下:
/**
* 长按邮箱快捷选项
* @author king
*/
private class quickwaylistener implements onitemlongclicklistener{
@override
public boolean onitemlongclick(adapterview<?> arg0, view view,
int pos, long id) {
//pos不可用说明见下文
return false;
}
}
如果这个方法是用在listview长按事件中刚刚好,但在expandablelistview中,第三个参数pos不能区分开点击的是父项还是子项,以及哪个父项或子项。
在expandablelistview响应的onitemlongckick方法中,pos参数值为:从上到下,父项+展现的子项到点击位置的数目(注意:是展现的,隐藏的子项不包括,从0开始)。
例如:
父项1(隐藏3个子项)
父项2
|—子项2-0
|—子项2-1
|—子项2-2
长按子项2-1时,pos值为3。显然根据pos值是无法确定点击的是哪个子项或父项的。
因此依赖pos是很难处理点击位置的。
如果可以直接在onitemlongclick方法中获取grouppos,及childpos该多好呢?
于是看到了onitemlongclick方法第二个参数:view。这里的view是你按中的位置对应的view。view有个方法gettag(int key)。如果在创建此view的时候就把grouppos,childpos通过settag(int key, object value)设置进去,在响应onitemlongclick不就可以直接拿出来用了么。
现在就要讲到必须使用自定义的baseexpandablelistadapter的理由了。
要把grouppos,childpos通过settag的方式绑定到view中,就必须操作该view的创建过程。要控制这个过程就必须要在自定义baseexpandablelistadapter中重写getgroupview及getchildview方法进行操作。如下:
复制代码 代码如下:
public class accountlistadapter extends baseexpandablelistadapter {
...省略其他方法
@override
public view getchildview(int groupposition, int childposition,
boolean islastchild, view convertview, viewgroup parent) {
//我这里仅通过自己写的mkchildview()方法创建textview来显示文字,更复杂的可以通过layoutinflater来填充一个view
textview childtv = mkchildview();
// 标记位置
// 必须使用资源id当key(不是资源id会出现运行时异常),android本意应该是想用tag来保存资源id对应组件。
// 将groupposition,childposition通过settag保存,在onitemlongclick方法中就可以通过view参数直接拿到了!
childtv.settag(r.id.xxx01, groupposition);
childtv.settag(r.id.xxx02, childposition);
return childtv;
}
@override
public view getgroupview(int groupposition, boolean isexpanded,
view convertview, viewgroup parent) {
textview grouptv = mkgroupview();
// 设置同getchildview一样
grouptv.settag(r.id.xxx01, groupposition);
grouptv.settag(r.id.xxx02, -1); //设置-1表示长按时点击的是父项,到时好判断。
grouptv.settext(groups[groupposition]);
return grouptv;
}
}
完成了这一步,我们只需要在expandablelistview响应的onitemlongclick方法时通过view.gettag(r.id.xxx01),view.gettag(r.id.xxx02)即可拿到grouppos,childpos.
如下:
复制代码 代码如下:
/**
* 长按邮箱快捷选项
* @author king
*/
private class quickwaylistener implements onitemlongclicklistener{
@override
public boolean onitemlongclick(adapterview<?> arg0, view view,
int pos, long id) {
int grouppos = (integer)view.gettag(r.id.xxx01); //参数值是在settag时使用的对应资源id号
int childpos = (integer)view.gettag(r.id.xxx02);
if(childpos == -1){//长按的是父项
//根据grouppos判断你长按的是哪个父项,做相应处理(弹框等)
} else {
//根据grouppos及childpos判断你长按的是哪个父项下的哪个子项,然后做相应处理。
}
return false;
}
}
到这就写完了,貌似比较啰嗦。重写baseexpandablelistadapter写的比较简洁,没看明白的朋友可以先到网上查下怎么自定义baseexpandablelistadapter,和自定义baseadapter其实是一样的。