• ListView事件的研究


    1. ListView的OnItemClickListener不被触发的另外一种情况

    image

    如上图,在一个ItemView中,只有一个TextView位于最左侧,他的右侧是空白区域,没有任何控件,当点击右侧区域时,并不会触发OnItemClickListener,当点击TextView所在的区域时,就能触发这个事件。

    看看这个事件的执行流程

    image

    右侧空白的部分没有View控件,也就是说虽然用手指点击了这一部分,但是没有view获取焦点,Android的事件触发是从顶层view一层层往下寻找的,如果有view获取焦点,就交给这个view处理,如果没有,就交给activity处理。

    click事件与touch事件的传播方式是不同的

    给ListView同时添加对touch和itemClick的监听事件,他们的触发顺序是:

    actionDown--------》action up---------》onItemClick

    他们的执行流程为:

    先按照touch事件的处理流程进行,然后在进行click事件的处理,这就说明,当用户按下屏幕,产生了两个事件,一个是touch事件,一个是click事件,添加到主线程的队列中。

    用户手指触摸屏幕,滚动ListView,看起来也进行了click操作,但是结果是,只触发了touch事件,没有触发click事件。

    2. ListView 获取焦点和ItemView获取焦点之间的关系

    2.1 ListView不获取焦点,ItemView能获取焦点吗?

    通 过设置ItemView的android:focusable="true"    android:focusableInTouchMode="true"属性,可以使ItemView在Touch mode 下获取焦点,默认情况下,Touch mode下ItemView,menu等等控件都是不能获取焦点的。只有ListView获取了焦点之后,ItemView才能获取焦点。

    实验一:

    设置ListView的focusable属性为true,ItemView的android:focusable="true"    android:focusableInTouchMode="true",自定义touch事件监听器,重写onTouch方法

    public boolean onTouch(View v, MotionEvent event) {
                switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    //clear();
                    int x = (int) event.getX();
                    int y = (int) event.getY();
                    int position = mListView.pointToPosition(x, y);
                    int firstVisiblePosition = mListView.getFirstVisiblePosition();
                    view = mListView.getChildAt(position-firstVisiblePosition);
                    if(view==null) return false;
                    if(view.isFocusable()){
                        Log.i(tag, "ItemView is focusable ");
                    }
                    if(view.isFocusableInTouchMode()){
                        Log.i(tag, "ItemView is focusable in touchMode");
                    }
                    if(view.isInTouchMode()){
                        Log.i(tag, "device is in touch mode.");
                    }
                    if(mListView.isFocusable()){
                        Log.i(tag, "mListView is focusable ");
                    }
                    if(mListView.isFocusableInTouchMode()){
                        Log.i(tag, "mListView isFocusableInTouchMode in touchMode");
                    }
                    
                    if(view.isFocused()){
                        Log.i(tag, "ItemView have get focus ");
                    }
                    if(mListView.isFocused()){
                        Log.i(tag, "mListView have get focus ");
                    }
                    if(view.isPressed()){
                        Log.i(tag, "ItemView have get pressed ");
                    }
                    break;
                case MotionEvent.ACTION_UP:
                    if(view==null) return false;
                    Log.i(tag, "OnTouchListener: up is working ");
                    if(view.isFocused()){
                        Log.i(tag, "ItemView have get focus ");
                    }
                    break;
                default:
                    break;
                }
                return false;
            }

    打印结果为:

    ItemView is focusable
    ItemView is focusable in touchMode
    device is in touch mode.
    mListView is focusable
    mListView isFocusableInTouchMode in touchMode
    mListView have get focus
    OnTouchListener: up is working

    当用户触摸屏幕的时候,触发了touch事件,但是只有ListView获取了焦点,itemView却没有获得焦点,说明itemView在默认状态下,即时设置了能获取焦点,能在touchmode下获取焦点,实际上也是不能的

    而 ListView的focusable属性即时不设为true,也是能够获取焦点的,那么如何让ItemView获取焦点呢,有两个方法,一个是View 类的requestFocus()方法,一个是ListView的requestChildFocus(child, focused)方法,还有就是requestFocusFromTouch()方法,该方法是View类的方法,ListView继承了该方法。

    我们让itemView调用requestFocus()方法或者requestFocusFromTouch()方法,这时itemView获取了焦点,但是ListView没有获取焦点,这说明一个视图中只能有一个view获取焦点。

    去掉itemView的android:focusableInTouchMode="true"属性,调用requestFocusFromTouch()方法,可以强制使itemViw获取焦点。ListView依旧没有获取焦点。

    总结:

    1. 当使用导航键上下左右滚动时,android框架会自动让view获取焦点(获取焦点后,就会高亮显示),然而当用户用手触摸屏幕的时候,就不需要让 view自动获取焦点了,也就是说,当用户点击了屏幕上的某个控件时,该控件就没有必要自动获取焦点了,因为用户知道自己操作的是哪个控件,当用户触摸手 机屏幕,就会进入touch mode模式。

    2. 在touch mode 模式下,有些控件是不会自动获取焦点的,但是还有些控件会,通过isFocusableInTouchMode()可以知道该控件在该touch模式下能 否获得焦点,TextView是默认不能获得焦点的,ListView默认能够获得焦点,EditText等文字编辑类控件都可以。通过设置 android:focusableInTouchMode="true"貌似可以使view控件获取焦点,但是实际上并不会是veiw控件获取焦点,还 需要手动调用requestFocusFromTouch()方法或者requestFocus()方法才能真正获取焦点。这里推荐使用 requestFocusFromTouch()方法,即时不用设置android:focusableInTouchMode="true",也能强制 使控件获取焦点。

    3. 获取ListView中有那个控件获取了焦点的方法

                    mListView.findFocus();
                    mListView.getFocusedChild();

    4. 通过ListView的setItemsCanFocus(true)方法并不可以使ItemView在touch mode下可以获取焦点,他只是表明在由ListAdapter创建的视图中,可包含能获得焦点的项目。

    滚 动事件发生在touch事件的后面,这种说法是不对的,通过实验可以得到onScroll方法的执行是在touch事件之前的,并且每一次触摸屏幕,先触 发这个方法,然后才触发touch事件,此外,当我们第一次进入列表界面时,onScroll方法也多次被调用,第一次是在执行onCreate方法时被 调用,这事还没有生成界面,所以visibleItemCount参数为0,然后的几次调用就有了界面了,visibleItemCount也被赋予界面 上显示的item的个数,显示不全的也算。差不多有两次,这两次的调用也是不一样的,看看这三次调用这个方法的不同

    图一:

    image

    图二:

    image

    图三:

    image

    当用手触摸屏幕上的某一项时,也会首先触发这个方法,然后才是touch事件

    image

    通过重写onTouch方法,可以实现当用户触摸屏幕的时候,itemView获取焦点,并且变色,但是出现了两个问题:

    1. 如果ListView可以显示多页,可以看到每页上又有一个ItemViw获取了焦点,不知道为什么?

    推测:Adapter中getView方法的第二个蚕食convertView是复用的,估计是生成新的页面的时候复用了获取焦点的veiw,这里还要看源码

    2. 当滚动到非第一页的时候,触发屏幕,并不能使触发点所在的ItemView获取焦点?

    ListView的getChildAt(int index)方法的参数index,与Adapter的getView方法的第一个参数position是不一样的,每个界面显示几个item,就建立从0到界面显示个数的索引,比如一个屏幕上显示8条记录,那么索引就是

    0---7,翻页了,仍然会建立类似的索引,因此应该计算出触发点的item在屏幕上的索引。

    int x = (int) event.getX();
    int y = (int) event.getY();
    int position = mListView.pointToPosition(x, y);
    int firstVisiblePosition = mListView.getFirstVisiblePosition();
    view = mListView.getChildAt(position-firstVisiblePosition);

  • 相关阅读:
    Django-序列化器
    HTTP请求头响应头的信息
    HTTP协议
    IAAS,PAAS,SAAS
    MySQL-部分名词解释
    MySQL-事务和特性
    用Java仿QQ界面聊天小项目
    25篇Java入门技术博客,送给准备入坑的你
    Java自学推荐书籍
    Java中的并发讲解
  • 原文地址:https://www.cnblogs.com/Free-Thinker/p/3396703.html
Copyright © 2020-2023  润新知