• Android实战技巧: ListView之ContextMenu无法弹出


    问题

    Activity中使用了ListView作为布局.当每一列表项中含有默认能获取焦点的子View时有可能会对ListView的某些事件有影响:

    1. OnItemClick
    2. OnItemLongClick
    3. ContextMenu

    这三个事件都无法正确响应.

    对于ContextMenu.首先要在onCreate中注册Activity的ContextMenu到ListView上:
    registerForContextMenu(mListView);
    要在onDestroy中取消注册:

    unregisterForContextMenu(mListView);

    一个简单的解法就是在Adapter中构建列表中的每个Item时,把相应事件Click或者LongClick加到具体的item上面.但这样的话,会造成逻辑十分混乱.而且点击Item或者长按Item时不会有Focus的Selection.此现象十分奇怪,必定另有隐情!

    Google了一下,果然StackOverflow上面有很多关于ListView 的ContextMenu的问题,首先要注意的是基本的用法,也就是上面提到的要registerForContextMenu.之后就是奇怪事件发生的地方了:

    如果ListView的列表项中含有默认就有Focus的子View(如Checkbox)时,就无法获取ContextMenu.

    这时就需要把Checkbox的Focus属性改变:

    android:focusable="false"

    然后就可以了.

    其实,ContextMenu也就是长按事件来的.所以不光是ContextMenu出不来,连onItemClick, onItemLongClick也都出不来!如果把Item中的Checkbox的android:focusable属性设置成false,就可以解决这些问题.

    但是,没有人能解释清楚为啥会酱紫!

    看了ListView的源代码,ListView并没有控制Click和LongClick以及ContextMenu(ContextMenu也是由长按事件触发的).这些事件是由它的父类AbsListView来管理的.
    在AbsListView中有一个用于检查LongClick的Runnable,它里面有这样一个判断:

    View v  = getChildAt(position);
    if (v != null && !v.hasFocusable()) {
          .... do long click handling which will trigger context menu.
    }


    也就是说每当长按一个项时,会检查其hasFocusable()返回值,返回false时才做长按的动作.其他地方比如onTouchEvent时也都会如此的检查,发现hasFocusable()为true时就直接返回!这里注意,检查的是所点击的列表项,所以如果列表项的布局不一样,现象就有可能不一样!
    那么View.hasFocusable(),当View的focusable为true时返回true,或者当其有子View的focusable为true时,就返回true.简单讲就是View本身focusable为true或者有子View的focusable为true时就返回true.
    因为,ListItem里面有CheckBox,它的focusable属性默认就是true的.所以就不会有ContextMenu弹出来.如果把CheckBox的focusable属性设置为false,就可以正常的弹出了.
    可以还是没弄明白为啥要酱紫设计?需要牛人来指点下!

    扩展

    目前来看默认就有focusable属性的有Button, CompoundButton, SeekBar, EditText,ImageButton,AutoCompleteTextView,WebView,WebTextView和它们的子类(怎么找出来了呢,到系统的style文件frameworks/base/core/res/res/value/styles.xml里面搜索focusable属性为true).所以理论上来讲,如果把这些Widget放入ListView里面时,ListView的OnItemClick,OnItemLongClick以及ContextMenu都不会有效果.解决办法就是把它们的focusable属性设为false.

  • 相关阅读:
    移动端小tips
    是否该放弃东莞的工作
    读书有感-learn html5 and javascript for ios
    eNSP多路由实现互联互通(华为路由E口直连)
    iTextSharp操作表格排版问题
    我今天开通博客
    12年的女程序员重新回归
    前端css样式规划
    前端神器-webstorm2017
    远程桌面资料共享
  • 原文地址:https://www.cnblogs.com/snake-hand/p/3148116.html
Copyright © 2020-2023  润新知