最近在项目中发现scrollview中嵌套listview会导致listview滚动失效,然后就上网查了一下资料,发现这个问题主要涉及到android事件的分发机制。要想很好的解决这个问题就一定要对android事件分发机制有一定的理解,通过网上查资料和查看android源码,对android事件分发机制有了一点自己的理解。
android的事件分发主要是action_down,action_move,action_up这些事件的传递和处理。而android主要是通过dispatchtouchevent,onintercepttouchevent,ontouchevent.这几个方法来实现事件的传递和处理。dispatchtouchevent传递和分发事件,ontouchevent处理事件,onintercepttouchevent拦截事件(控制事件是否向子控件传递)。
当我们点击某一个控件时会触发action_down,action_up这两个事件。由于action_down事件的传递和其它事件的传递不一样,这边先来看action_down事件的传递,默认的流程如下所示:
上面这个只是默认情况下的传递流程图,我们可以看到android的事件传递大概是一个U型结构,而listview的滑动失效通过日志记录发现action_down是能够传递到listview的ontouchevent中的,但是action_move事件在传递到scrollview的dispatchtouchevent中时由于onintercepttouchevent返回true,传递到listview的事件变为action_cancel事件,导致listview无法滑动。解决的方案也很简单我们只要重写scrollview的onintercepttouchevent事件返回false就可以了,但是这样也会有新的问题就是在scrollview的所有子控件上滑动的时候scrollview都不会再滚动,要解决这个就需要我们在viewgroup的onintercepttouchevent中判断listview的区域来决定返回false还是super.onintercepttouchevent().