ClickableSpan形成Listview的OnItemClickListener失效的解決辦法

1、前提和解決
作了個界面,在listview的itemview裏 要@,要超連接,要話題跳轉等等等。
用ClickableSpan實現了textview的點擊跳轉,以後發現listview的OnItemClickListener不響應,給textview的focusable設置爲false,或者listview的descendantFocusability爲blocksDescendants都無論使。
百度後發現了Terry_龍的代碼看着很科學,但是複製粘貼後發現仍是不行,繼續百度找到一個求助貼,接着跳轉到了一個英文網站。。介紹完艱辛旅程,下面是正解:html

/* ********************** 解決方法  ************************ */
一、抄完Terry_龍的代碼java

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);
    }
}
}

二、在自定義的TextView多寫一個 android

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


三、完工,  隨後你就可使用 ide

setMovementMethod(TextViewFixTouchConsume.LocalLinkMovementMethod.getInstance());

這樣即給TextView增長點擊效果,又不讓其佔用Item的點擊焦點。相似微博的@ 、表情、連接等。測試


/* ********************** 解決方法到此結束  ************************ */
網站


2、原理 (來自 英文網頁):
spa

當調用textview的setMovementMethod 或者 setKeyListener, TextView 自動修改它的屬性:code

 setFocusable(true);

一、這也就是說你手動設置的focusable被覆蓋掉了,也就須要咱們覆寫hasFocusable方法,使其始終返回false。htm

二、覆寫hasFocusable以後listview的OnItemClick已經能夠響應,以有限的大腦容量我覺得自定義的LinkMovementMethod能夠不要了,惋惜註釋掉以後發現ClickableSpan不工做了。。。

3、已知Bug:
在上面找到的求助貼裏樓主問:使用這個解決方案後每次滑動都會調用onItemLongClick事件,我又測試了一下把長按事件加在TextView上,發現同樣會在滑動時被激活。就是說使用這個方法後長按事件就別用了。

參考: blog

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

相關文章
相關標籤/搜索