问题引出:ListView 的 Item 有Button等可点击控件的时候,Button设置了onClickListener后,上层的Item能不能接收到点击事件?
ViewGroup包含子View时,存在焦点争夺,默认情况下,子view中含有可点击控件时(Button)会获取到焦点, ViewGroup将事件往子view分发时,如果ViewGroup没有拦截(没焦点的情况下是否拦截在源码中尚未分析出来,测试listview的实例中是没有拦截的),事件会传递至点击位置所对应的子view,子view拥有焦点并且有OnClickListener监听,则在子view的onTouchEvent中回调onclick函数,onclick函数是消耗事件的,所以上层的ViewGroup(item)无法再接受到事件。
如果点击位置没有子View(点击item上的空白),则事件回到ViewGroup(item)的onTouchEvent函数中,同时只有持有焦点的View才能响应其设置的OnClickListener监听(View源码10315行),此时子view拥有焦点,所以item不回调onclick函数。
如果想让item和button都能响应各自的点击事件,可以在子view布局文件中设置:android:focusable=”true”,或代码设置view.setFocusable(false);或者不改变子view的默认焦点,在item布局文件中设置:android:descendantFocusability=”blocksDescendants”(ViewGroup优先获取焦点并且屏蔽子view焦点),或代码设置viewGroup.setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS);
一句话,点击监听处于事件分发传递的末级,在各view的onTouchEvent中调用,受焦点的影响。
注:ViewGroup子view中存在焦点,则ViewGroup去requesFocus也拿不到焦点,View源码中也有类似操作,在onTouchEvent函数中,触摸事件传递到当前位置,如果当前view没有焦点会先去获取焦点,如果是ViewGroup则遍历子View,子view有焦点则ViewGroup获取不到焦点。获取不到焦点的view无法回调onclick函数。
关于焦点对于事件的传递和点击监听的影响(个人观点,只影响监听的回调不影响事件分发流程),由于看的时间较短,上诉分析可能存在错误,欢迎指出。