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

原创
2013/07/17 09:47
阅读数 6.1K

一、前提和解决
做了个界面,在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

展开阅读全文
打赏
0
3 收藏
分享
加载中
丁佼博主

引用来自“kkmike999”的评论

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

引用来自“张滔”的评论

引用来自“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事件又不行了
2014/03/03 14:01
回复
举报

引用来自“张滔”的评论

引用来自“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;
  }
不知道你有什么解决方式没有?
2014/03/03 10:48
回复
举报

引用来自“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;

  }
2014/03/01 13:45
回复
举报

引用来自“张滔”的评论

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

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

引用来自“niuky”的评论

已经解决

2求方案。有链接么?
2014/01/25 13:03
回复
举报
已经解决
2014/01/23 15:27
回复
举报
更多评论
打赏
13 评论
3 收藏
0
分享
返回顶部
顶部