文档章节

ClickableSpan造成Listview的OnItemClickListener失效的解决办法

丁佼
 丁佼
发布于 2013/07/17 09:47
字数 633
阅读 5.8K
收藏 3

一、前提和解决
做了个界面,在listview的itemview里 要@,要超链接,要话题跳转等等等。
用ClickableSpan实现了textview的点击跳转,之后发现listview的OnItemClickListener不响应,给textview的focusable设置为false,或者listview的descendantFocusability为blocksDescendants都不管使。
百度后发现了Terry_龙的代码看着很科学,可是复制粘贴后发现还是不行,继续百度找到一个求助贴,接着跳转到了一个英文网站。。介绍完艰辛旅程,下面是正解:

/* ********************** 解决方法  ************************ */
1、抄完Terry_龙的代码

public class TextViewFixTouchConsume extends TextView {
boolean dontConsumeNonUrlClicks = true;
boolean linkHit;

public TextViewFixTouchConsume(Context context) {
    super(context);
}

public TextViewFixTouchConsume(Context context, AttributeSet attrs) {
    super(context, attrs);
}

public TextViewFixTouchConsume(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
}

@Override
public boolean onTouchEvent(MotionEvent event) {
    linkHit = false;
    boolean res = super.onTouchEvent(event);

    if (dontConsumeNonUrlClicks)
        return linkHit;
    return res;

}

public static class LocalLinkMovementMethod extends LinkMovementMethod{
    static LocalLinkMovementMethod sInstance;


    public static LocalLinkMovementMethod getInstance() {
        if (sInstance == null)
            sInstance = new LocalLinkMovementMethod();

        return sInstance;
    }

    @Override
    public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) {
        int action = event.getAction();

        if (action == MotionEvent.ACTION_UP ||
                action == MotionEvent.ACTION_DOWN) {
            int x = (int) event.getX();
            int y = (int) event.getY();

            x -= widget.getTotalPaddingLeft();
            y -= widget.getTotalPaddingTop();

            x += widget.getScrollX();
            y += widget.getScrollY();

            Layout layout = widget.getLayout();
            int line = layout.getLineForVertical(y);
            int off = layout.getOffsetForHorizontal(line, x);

            ClickableSpan[] link = buffer.getSpans(off, off, ClickableSpan.class);

            if (link.length != 0) {
                if (action == MotionEvent.ACTION_UP) {
                    link[0].onClick(widget);
                } else if (action == MotionEvent.ACTION_DOWN) {
                    Selection.setSelection(buffer,
                            buffer.getSpanStart(link[0]),
                            buffer.getSpanEnd(link[0]));
                }

                if (widget instanceof TextViewFixTouchConsume){
                    ((TextViewFixTouchConsume) widget).linkHit = true;
                }
                return true;
            } else {
                Selection.removeSelection(buffer);
                Touch.onTouchEvent(widget, buffer, event);
                return false;
            }
        }
        return Touch.onTouchEvent(widget, buffer, event);
    }
}
}

2、在自定义的TextView多写一个

@Override
public boolean hasFocusable() {
   return false;
}


3、完工,  随后你就可以使用

setMovementMethod(TextViewFixTouchConsume.LocalLinkMovementMethod.getInstance());

这样即给TextView增加点击效果,又不让其占用Item的点击焦点。类似微博的@ 、表情、链接等。


/* ********************** 解决方法到此结束  ************************ */


二、原理 (来自 英文网页):

当调用textview的setMovementMethod 或者 setKeyListener, TextView 自动修改它的属性:

 setFocusable(true);

1、这也就是说你手动设置的focusable被覆盖掉了,也就需要我们覆写hasFocusable方法,使其始终返回false。

2、覆写hasFocusable之后listview的OnItemClick已经可以响应,以有限的大脑容量我以为自定义的LinkMovementMethod可以不要了,可惜注释掉之后发现ClickableSpan不工作了。。。

三、已知Bug:
在上面找到的求助贴里楼主问:使用这个解决方案后每次滑动都会调用onItemLongClick事件,我又测试了一下把长按事件加在TextView上,发现一样会在滑动时被激活。就是说使用这个方法后长按事件就别用了。

参考: 

链接一、http://www.cnblogs.com/TerryBlog/archive/2013/04/02/2994815.html
链接二、http://www.eoeandroid.com/thread-275209-1-1.html
链接三、http://stackoverflow.com/questions/8558732/listview-textview-with-linkmovementmethod-makes-list-item-unclickable

© 著作权归作者所有

丁佼
粉丝 5
博文 26
码字总数 9223
作品 0
连云港
程序员
私信 提问
加载中

评论(13)

丁佼
丁佼 博主

引用来自“kkmike999”的评论

为了解决Clickable引起的各种冲突,还不如用onTouch从根本上解决~
自己计算文字长度、点击位置、点击内容类型,看似活不多,不过写出来BUG应该蛮多的。。。不是本科生咱就不操那个心了。系统控件用着安心。。。
kkmike999
kkmike999
为了解决Clickable引起的各种冲突,还不如用onTouch从根本上解决~
niuky
niuky

引用来自“张滔”的评论

引用来自“niuky”的评论

引用来自“张滔”的评论

在Ontouch事件中加入下面这段,完美解决    if(action == MotionEvent.ACTION_DOWN){
          try {
            Thread.sleep(200);
          } catch (InterruptedException e) {
          }
        }

这个是解决长按问题还是解决滑动listview 误触发@这些过滤内容的点击?

这只是暂时解决了长按问题,但滑动的时候如果手指碰到TextView还是会触发长按事件,经过一番琢磨,发现在这样子才是最完美的,将textView 的onTouch事件改成
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    linkHit = false;
    boolean res = super.onTouchEvent(event);
    if (dontConsumeNonUrlClicks) {
        if(!linkHit){
          if(isLongClickable()){
            return true;
        }
      }
      return linkHit;
    }
    return res;

  }

你这种暂时是解决了长按问题,但是itemclick事件又不行了
niuky
niuky

引用来自“张滔”的评论

引用来自“niuky”的评论

引用来自“张滔”的评论

在Ontouch事件中加入下面这段,完美解决    if(action == MotionEvent.ACTION_DOWN){
          try {
            Thread.sleep(200);
          } catch (InterruptedException e) {
          }
        }

这个是解决长按问题还是解决滑动listview 误触发@这些过滤内容的点击?

这只是暂时解决了长按问题,但滑动的时候如果手指碰到TextView还是会触发长按事件,经过一番琢磨,发现在这样子才是最完美的,将textView 的onTouch事件改成
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    linkHit = false;
    boolean res = super.onTouchEvent(event);
    if (dontConsumeNonUrlClicks) {
        if(!linkHit){
          if(isLongClickable()){
            return true;
        }
      }
      return linkHit;
    }
    return res;

  }

看到网上用的这个方法 不过测试了下会有一个问题 就是说滑动列表的时候碰到textview会触发@ 话题等相关的点击事件
  @Override
  public boolean onTouchEvent(MotionEvent event) {

    int action = event.getAction();
    boolean flag = false;
    switch (action) {
    case MotionEvent.ACTION_DOWN:
      MovementMethod m = getMovementMethod();
      setMovementMethod(null);
      Log.d("onTouchEvent", getText().toString());
      boolean mt = m.onTouchEvent(this, (Spannable) getText(), event);
      if (mt) {
        event.setAction(MotionEvent.ACTION_UP);
        mt = m.onTouchEvent(this, (Spannable) getText(), event);
        event.setAction(MotionEvent.ACTION_DOWN);
      }
      boolean st = super.onTouchEvent(event);
      setMovementMethod(m);
      setFocusable(false);
      flag = mt || st;
      break;
    default:
      break;
    }
    return flag;
  }
不知道你有什么解决方式没有?
张滔
张滔

引用来自“niuky”的评论

引用来自“张滔”的评论

在Ontouch事件中加入下面这段,完美解决    if(action == MotionEvent.ACTION_DOWN){
          try {
            Thread.sleep(200);
          } catch (InterruptedException e) {
          }
        }

这个是解决长按问题还是解决滑动listview 误触发@这些过滤内容的点击?

这只是暂时解决了长按问题,但滑动的时候如果手指碰到TextView还是会触发长按事件,经过一番琢磨,发现在这样子才是最完美的,将textView 的onTouch事件改成
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    linkHit = false;
    boolean res = super.onTouchEvent(event);
    if (dontConsumeNonUrlClicks) {
        if(!linkHit){
          if(isLongClickable()){
            return true;
        }
      }
      return linkHit;
    }
    return res;

  }
niuky
niuky

引用来自“张滔”的评论

在Ontouch事件中加入下面这段,完美解决    if(action == MotionEvent.ACTION_DOWN){
          try {
            Thread.sleep(200);
          } catch (InterruptedException e) {
          }
        }

这个是解决长按问题还是解决滑动listview 误触发@这些过滤内容的点击?
张滔
张滔
在Ontouch事件中加入下面这段,完美解决    if(action == MotionEvent.ACTION_DOWN){
          try {
            Thread.sleep(200);
          } catch (InterruptedException e) {
          }
        }
niuky
niuky
http://ask.csdn.net/questions/2807
丁佼
丁佼 博主

引用来自“niuky”的评论

已经解决

2求方案。有链接么?
niuky
niuky
已经解决
ClickableSpan造成Listview的OnItemClickListener失效的解决办法

@丁佼 你好,想跟你请教个问题:我按照你说的方法,可还是失效,不能点击。不知道是怎么回事

hf496486126
2013/09/24
170
1
Android 开发中踩过的坑之十二: ListView中的焦点抢夺

当ListView设置了OnItemClickListener, Item项里的View设置了OnClickListener时, 经常遇到某一个Listener失效的现象. 根本原因是焦点的问题, 当item项目中某个View能够获取焦点时, 根据View焦...

revfycd
2015/10/16
616
2
由ListView改变Item背景问题引出的对自定义Adapter的反思

由ListView改变Item背景问题引出的对自定义Adapter的反思 本人新手,前两天做关于ListView的一个信息栏,想要做出一种点击选中改变背景图片的效果,遇到了不少麻烦,在网上也没有找到一个完整...

华宰
2011/09/20
1.1K
1
ListView在开发中的小技巧

如何取消Listview的滚动条? setVerticalScrollBarEnabled(false) 2.白色的背景,ListView滚屏进行中的时候,背景会变成黑色,解决办法? android:cacheColorHint="#00000000" 3.ListView滚动......

紫地瓜
2013/01/15
168
0
ListView不响应事件的解决办法

如果你的自定义ListViewItem中有Button或者Checkable的子类控件的话,那么默认focus是交给了子控件,而 ListView的Item能被选中的基础是它能获取Focus,也就是说我们可以通过将ListView中Ite...

hanQ
2012/09/28
212
0

没有更多内容

加载失败,请刷新页面

加载更多

上海哪里可以开服务费发票-上海新闻网

上海哪里可以开服务费发票【1.3.2 - 2.9.3.0 - 0.5.6.8.】李生,adb的全称为Android Debug Bridge,是Android手机通用的一个USB端口。百度CarLife的部分车机采用了...

多徐重
7分钟前
25
0
上海哪里可以开加油费发票-上海新闻网

上海哪里可以开加油费发票【1.3.2 - 2.9.3.0 - 0.5.6.8.】李生,adb的全称为Android Debug Bridge,是Android手机通用的一个USB端口。百度CarLife的部分车机采用了...

上感受到
7分钟前
25
0
上海哪里可以开住宿费发票-上海新闻网

上海哪里可以开住宿费发票【1.3.2 - 2.9.3.0 - 0.5.6.8.】李生,adb的全称为Android Debug Bridge,是Android手机通用的一个USB端口。百度CarLife的部分车机采用了...

识过人石
9分钟前
65
0
上海哪里可以开建筑业发票-上海新闻网

上海哪里可以开建筑业发票【1.3.2 - 2.9.3.0 - 0.5.6.8.】李生,adb的全称为Android Debug Bridge,是Android手机通用的一个USB端口。百度CarLife的部分车机采用了...

提供格
10分钟前
29
0
OSChina 周一乱弹 —— 提前接受社会的毒打教育

Osc乱弹歌单(2020)请戳(这里) 【今日歌曲】 @爱吃蛋挞的kk:分享Buddy Guy的单曲《I Need You Tonight》 《I Need You Tonight》- Buddy Guy 手机党少年们想听歌,请使劲儿戳(这里) 凌...

小小编辑
38分钟前
46
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部