ListView中嵌入布局的Button或多个点击事件
有时候在ListView嵌入的布局中有多个事件需要点击,比如一个item中有TextView和Button两个布局,当我们需要获取这两个点击事件时,我们应该如何去获取呢,通常来说,我们都是已经固定好了TextView和Button的id的,,所以,这两个点击事件的id肯定是不会变的,假如我们有10个item,我们需要获取第5个item中的布局的button和TextView的点击事件该如何获取呢?下面将依依介绍。(适合初学者)
首先先看初次布局界面:
activity_main.xml布局:ListView的布局
- <?xml version="1.0" encoding="utf-8"?>
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- tools:context="com.qianfeng.listviewbutton.MainActivity">
- <ListView
- android:id="@+id/mList"
- android:layout_width="match_parent"
- android:layout_height="match_parent">
- </ListView>
- </RelativeLayout>
list_item.xml布局:ListView item的每个布局
- <?xml version="1.0" encoding="utf-8"?>
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="horizontal" android:layout_width="match_parent"
- android:layout_height="match_parent">
- <TextView
- android:id="@+id/mTv"
- android:layout_marginTop="10dp"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textSize="20sp"
- android:text="234"
- />
- <Button
- android:id="@+id/mBtn"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_alignParentRight="true"
- />
- </RelativeLayout>
先看我们的初始代码:
MainActivity.java
- public class MainActivity extends AppCompatActivity {
- //ListView控件
- private ListView mList;
- //ListView数据源
- private List<String> data;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- data = new ArrayList<>();
- mList = (ListView)findViewById(R.id.mList);
- for(int i = 0; i < 20; i ++){
- data.add("今天好手气" + i);
- }
- MyAdapter adapter = new MyAdapter(data);
- mList.setAdapter(adapter);
- //ListView item点击事件
- mList.setOnItemClickListener(new ListView.OnItemClickListener(){
- @Override
- public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
- Toast.makeText(MainActivity.this,"我是item点击事件 i = " + i + "l = " + l,Toast.LENGTH_SHORT).show();
- }
- });
- }
- }
MyAdapter.java
- public class MyAdapter extends BaseAdapter implements View.OnClickListener {
- //上下文
- private Context context;
- //数据项
- private List<String> data;
- public MyAdapter(List<String> data){
- this.data = data;
- }
- @Override
- public int getCount() {
- return data == null ? 0 : data.size();
- }
- @Override
- public Object getItem(int i) {
- return data.get(i);
- }
- @Override
- public long getItemId(int i) {
- return i;
- }
- @Override
- public View getView(int i, View view, ViewGroup viewGroup) {
- ViewHolder viewHolder = null;
- if(context == null)
- context = viewGroup.getContext();
- if(view == null){
- view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.list_item,null);
- viewHolder = new ViewHolder();
- viewHolder.mTv = (TextView)view.findViewById(R.id.mTv);
- viewHolder.mBtn = (Button)view.findViewById(R.id.mBtn);
- view.setTag(viewHolder);
- }
- //获取viewHolder实例
- viewHolder = (ViewHolder)view.getTag();
- //设置数据
- viewHolder.mTv.setText(data.get(i));
- //设置监听事件
- viewHolder.mTv.setOnClickListener(this);
- //设置数据
- viewHolder.mBtn.setText("点我点我"+ i);
- viewHolder.mBtn.setOnClickListener(this);
- return view;
- }
- @Override
- public void onClick(View view) {
- switch (view.getId()){
- case R.id.mBtn:
- Log.d("tag", "Btn_onClick: " + "view = " + view);
- Toast.makeText(context,"我是按钮",Toast.LENGTH_SHORT).show();
- break;
- case R.id.mTv:
- Log.d("tag", "Tv_onClick: " + "view = " + view);
- Toast.makeText(context,"我是文本",Toast.LENGTH_SHORT).show();
- break;
- }
- }
- static class ViewHolder{
- TextView mTv;
- Button mBtn;
- }
- }
通过上图我们可以看到,当鼠标第一次停留的地方,我点击item项时,没有任何反应,当点击按钮和文本的时候就会弹出吐司,那么为什么点击item会没有触发事件呢?怎样才能让item也有触发事件呢?
因为一个item布局中有按钮控件的话,按钮会获得焦点,而此时item就获取不到焦点,所以点击Item时不能触发其点击事件,如果想让item也有点击事件的话,则设置Button的焦点默认为false,则我们只需要在list_item.xml中的Button控件中加入android:focusable="false"
这样item就有了焦点,可以点击了
如下是改变焦点后的运行效果:
但是随之又有一个问题了,我们如何判断我们点击的是哪个item中的TextView和Button呢,因为目前的点击时间只能判断是TextView点击了,还是Button,或者item被点击了,除了item能知道是哪一项被点击了,其他两个却不知道是在哪个item中被点击了,所以我们需要将代码进行在此修改。只需要在MyAdapter中修改代码即可。基本思路是,我们可以将每个被点击的控件中设置一个标记,通过View中的setTag(int key, Object tag)方法设置即可,第一个key必须是一个资源id
资源id的添加
ids.xml
- <?xml version="1.0" encoding="utf-8"?>
- <resources>
- <item name="btn" type="id"></item>
- <item name="tv" type="id"></item>
- </resources>
MyAdapter.java
- public class MyAdapter extends BaseAdapter implements View.OnClickListener {
- private Context context;
- private List<String> data;
- public MyAdapter(List<String> data){
- this.data = data;
- }
- @Override
- public int getCount() {
- return data == null ? 0 : data.size();
- }
- @Override
- public Object getItem(int i) {
- return data.get(i);
- }
- @Override
- public long getItemId(int i) {
- return i;
- }
- @Override
- public View getView(int i, View view, ViewGroup viewGroup) {
- ViewHolder viewHolder = null;
- if(context == null)
- context = viewGroup.getContext();
- if(view == null){
- view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.list_item,null);
- viewHolder = new ViewHolder();
- viewHolder.mTv = (TextView)view.findViewById(R.id.mTv);
- viewHolder.mBtn = (Button)view.findViewById(R.id.mBtn);
- view.setTag(viewHolder);
- }
- viewHolder = (ViewHolder)view.getTag();
- //设置tag标记
- viewHolder.mBtn.setTag(R.id.btn,i);//添加此代码
- viewHolder.mBtn.setText("点我点我"+ i);
- viewHolder.mBtn.setOnClickListener(this);
- viewHolder.mTv.setText(data.get(i));
- //设置tag标记
- viewHolder.mTv.setTag(R.id.tv,i);//添加此代码
- viewHolder.mTv.setOnClickListener(this);
- return view;
- }
- @Override
- public void onClick(View view) {
- switch (view.getId()){
- case R.id.mBtn:
- int b = (int) view.getTag(R.id.btn);
- Toast.makeText(context,"我是按钮 " + b,Toast.LENGTH_SHORT).show();
- break;
- case R.id.mTv:
- int t = (int)view.getTag(R.id.tv);
- Toast.makeText(context,"我是文本" + t,Toast.LENGTH_SHORT).show();
- break;
- }
- }
- static class ViewHolder{
- TextView mTv;
- Button mBtn;
- }
- }
基本已经实现了,案例比较简单,适合初学者。
如果你是用ImageView,还是会出现item点击无效,接着看下面:
Android开发中在ListView中经常有Item的点击事件及Button或ImageButton等需要被点击的事件之间的处理方法
解决的办法,在ListView的Item的xml文件中添加如下属性:
1)整个xml文件的根元素如LinearLayout中添加属性android:descendantFocusability="blocksDescendants"
2)被点击的控件如ImageButton中添加属性android:focusable="false"和android:clickable="true"
开发中很常见的一个问题,项目中的listview不仅仅是简单的文字,常常需要自己定义listview,自己的Adapter去继承BaseAdapter,在adapter中按照需求进行编写,问题就出现了,可能会发生点击每一个item的时候没有反应,无法获取的焦点。原因多半是由于在你自己定义的Item中存在诸如ImageButton,Button,CheckBox等子控件(也可以说是Button或者Checkable的子类控件),此时这些子控件会将焦点获取到,所以常常当点击item时变化的是子控件,item本身的点击没有响应。
这时候就可以使用descendantFocusability来解决啦,API描述如下:
android:descendantFocusability
Defines the relationship between the ViewGroup and its descendants when looking for a View to take focus.
Must be one of the following constant values.
该属性是当一个为view获取焦点时,定义viewGroup和其子控件两者之间的关系。
属性的值有三种:
beforeDescendants:viewgroup会优先其子类控件而获取到焦点
afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点
blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点
通常我们用到的是第三种,即在Item布局的根布局加上android:descendantFocusability=”blocksDescendants”的属性就好了,至此listview点击的灵异事件告一段落。心得:遇到不会不懂的地方除了网上查询资料之外,也可以多多去尝试每种属性的作用,多阅读官方文档(我始终觉得还是读原文的比翻译的理解的会更好)。
ListView中嵌入布局的Button或多个点击事件的更多相关文章
- 解决ListView中Item的子控件与Item点击事件冲突
常常会碰到在ListView中点击当中一个Item.会一并触发其子控件的点击事件.比如Item中的Button.ImageButton等.导致了点击Item中Button以外区域也会触发Button点 ...
- android ListView中的Item有Button时候点击异常处理
1.当ListView中有Button的时候往往会遇到很多问题,比较常见的一个问题是: 假设:在ListView中有N个Item当点击其中某个Item中的Button的时候,需要改变当前Button的 ...
- 【转】Android - Button(按钮)的响应点击事件的4种写法
原文网址:http://www.yrom.net/blog/2011/12/12/android-4-onclicklistener-of-button/ Button控件setOnclickList ...
- 注意:MainActivity的oncreate方法里不要再inflate布局了(MainActivity里的点击事件无响应)
activity_main已经通过setContentView(R.layout.activity_main);设置给MainActivity, 不要再inflate出新布局,然后findviewby ...
- Android - Button(按钮)的响应点击事件的4种写法
Button控件setOnclickListener(View.OnClickListener listener)来接收一个点击事件的监听器 自定义一个点击事件监听器类让其实现View.OnClick ...
- Button控件的点击事件
Java的代码 主要重点: findViewById(); OnClickListener() package com.example.admin.myapplication; import andr ...
- js中嵌入jsp(html)代码的双引号转换问题--事件没反应
下面是一段今天遇到问题的代码,select中写了onchange事件 ,在没有加转义的情况下,F12解析的代码是错乱的,双引号与内容中写的不一致,还会有空格出现,经过一段时间的摸索,发现在出错的地方加 ...
- ListView item中有button或EditText时 点击事件失效问题的解决
加入:android:descendantFocusability="blocksDescendants" 代码如下: <LinearLayout xmlns:android ...
- ListView中多个EditText设置焦点 多次点击异常报错
08-17 18:23:09.825: ERROR/AndroidRuntime(1608): FATAL EXCEPTION: main 08-17 18:23:09.825: ERROR/Andr ...
随机推荐
- Nginx 配置文件nginx.conf中文详解
######Nginx配置文件nginx.conf中文详解##### #定义Nginx运行的用户和用户组 user www www; #nginx进程数,建议设置为等于CPU总核心数. worker_ ...
- 实验3 SQL注入原理-万能密码注入
实验目的 (1)理解[万能密码]的原理 (2)学习[万能密码]的使用 实验原理 一.访问目标网站 1.选择一个存在漏洞的论坛 http://192.168.1.3:8009 进入 2.输入用户名[ad ...
- docker里运行cron的要点笔记
1.如果用精简的apline或debian:stretch-slim创建的docker,里面可能没有cron模块,需要独立安装 apt-get install -y cron 2.docker里面cr ...
- 日常bug(1)
今天在写写代码的时候,前端找我,说我写的一个接口有可能有问题.导致前端的数据不能正常显示,我去看了一下,确实不能正常显示.问题的原因是本来前端循环遍历一个json里的数组,但是接下来的数据变成对象了. ...
- hdu3486Interviewe(二分是错的)(ST算法RMQ + 判定上下界枚举)
题目大意是找最小的m使得前m段中每一段的最大值相加严格大于k,每一段长度为[n/m](n/m向下取整,多余的后半部分部分n-m*[n/m]不要) 先给一段我一开始的思路,和网上许多题解思路一样,但其实 ...
- 《exception》第九次团队作业:Beta冲刺与验收准备(第二天)
一.项目基本介绍 项目 内容 这个作业属于哪个课程 任课教师博客主页链接 这个作业的要求在哪里 作业链接地址 团队名称 Exception 作业学习目标 1.掌握软件黑盒测试技术:2.学会编制软件项目 ...
- 各大公司Java面试题收录含答案(整理版)持续中....
本文分为17个模块,分别是:Java基础.容器.多线程.反射.对象拷贝.Java web.异常.网络.设计模式.算法.Spring/Spring MVC.Spring Boot/Spring Clou ...
- 微信小程序——通讯录
WXML: <view class="flex box box-lr"> <scroll-view class="flex groups box box ...
- BZOJ 2200: [Usaco2011 Jan]道路和航线
Description Farmer John正在一个新的销售区域对他的牛奶销售方案进行调查.他想把牛奶送到T个城镇 (1 <= T <= 25,000),编号为1T.这些城镇之间通过R条 ...
- 04-Flutter移动电商实战-打通底部导航栏
关于界面切换以及底栏的实现可参考之前写的一篇文章:Flutter实 ViewPager.bottomNavigationBar界面切换 1.新建4个基本dart文件 在pages目录下,我们新建下面四 ...