Android事件系统(2)ClickableSpan In Clickable TextView

ClickableSpan In Clickable TextView

我在使用ClickableSpan时想实现以下功能,点击ClickableSpan进入事件1,点击TextView中剩余部分进入事件2,不幸的是,我发现在默认情况下,点击ClickableSpan区域会同时触发TextViewOnClick事件和ClickableSpanonClick事件,而且在响应时,明显感觉到TextViewonClick事件比ClickableSpanonClick事件响应稍迟,这是因为TextViewonClick事件在一个新的线程响应,而ClickableSpan的事件直接在UI线程响应,接下来通过查看TextView源码我发现了这种冲突的解决方案。
TextViewonTouchEvent源码如下:

public boolean onTouchEvent(MotionEvent event) {
        final int action = event.getActionMasked();

       if (mEditor != null) mEditor.onTouchEvent(event);

        final boolean superResult = super.onTouchEvent(event);

        /*
         * Don't handle the release after a long press, because it will
         * move the selection away from whatever the menu action was
         * trying to affect.
         */
        if (mEditor != null && mEditor.mDiscardNextActionUp && action == MotionEvent.ACTION_UP) {
            mEditor.mDiscardNextActionUp = false;
            return superResult;
        }

        final boolean touchIsFinished = (action == MotionEvent.ACTION_UP) &&
                (mEditor == null || !mEditor.mIgnoreActionUpEvent) && isFocused();

         if ((mMovement != null || onCheckIsTextEditor()) && isEnabled()
                && mText instanceof Spannable && mLayout != null) {
           boolean handled = false;

            if (mMovement != null) {
                handled |= mMovement.onTouchEvent(this, (Spannable) mText, event);
            }

            final boolean textIsSelectable = isTextSelectable();
            if (touchIsFinished && mLinksClickable && mAutoLinkMask != 0 && textIsSelectable) {
                // The LinkMovementMethod which should handle taps on links has not been installed
                // on non editable text that support text selection.
                // We reproduce its behavior here to open links for these.
               ClickableSpan[] links = ((Spannable) mText).getSpans(getSelectionStart(),
                        getSelectionEnd(), ClickableSpan.class);

               if (links.length > 0) {
                    links[0].onClick(this);
                    handled = true;
                }
            }

            if (touchIsFinished && (isTextEditable() || textIsSelectable)) {
                // Show the IME, except when selecting in read-only text.
                final InputMethodManager imm = InputMethodManager.peekInstance();
                viewClicked(imm);
                if (!textIsSelectable && mEditor.mShowSoftInputOnFocus) {
                    handled |= imm != null && imm.showSoftInput(this, 0);
                }

                // The above condition ensures that the mEditor is not null
                mEditor.onTouchUpEvent(event);

                handled = true;
            }

            if (handled) {
                return true;
            }
        }

        return superResult;
    }

从以上源码我们可以看出在点击事件发生时,TextView首先调用了super.onTouchEvent(event)触发了TextViewonClick事件,此时并没有结束事件,而是一路向下,又来到了links[0].onClick(this);触发了ClickableSpanonClick事件,进而两个监听事件都被执行了。搞清楚原因之后我们就来处理,重写TextViewonTouchEvent事件,代码如下:

import android.content.Context;
import android.content.Intent;
import android.text.Spannable;
import android.text.method.MovementMethod;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

/**
 * Created by TuoZhaoBing on 2016/4/28 0028.
 */
public class ClickPreventableTextView extends TextView implements View.OnClickListener {
    private boolean preventClick;
    private View.OnClickListener clickListener;
    private boolean ignoreSpannableClick;

    public void setIgnoreSpannableClick(boolean ignoreSpannableClick) {
        this.ignoreSpannableClick = ignoreSpannableClick;
    }

    public Node mNode;
    public RootNode mRoot;
    private Context mContext;
    private MindGraph mVg;

    public ClickPreventableTextView(Context context, MindGraph vg) {
        super(context);
        setClickable(true);
        mContext = context;

        mVg = vg;
        setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (null != mNode){
                    Intent intent = new Intent(mContext,Main2Activity.class);
                    mContext.startActivity(intent);
                }
            }
        });
    }

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

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

    public boolean onTouchEvent(MotionEvent event) {
        if (getMovementMethod() != null)
            ((ClickableMovementMethod)getMovementMethod()).onTouchEvent(this, (Spannable)getText(), event);
        this.ignoreSpannableClick = true;
        boolean ret = super.onTouchEvent(event);
        this.ignoreSpannableClick = false;
        return ret;
    }

    /**
     * Returns true if click event for a clickable span should be ignored
     * @return true if click event should be ignored
     */
    public boolean ignoreSpannableClick() {
        return ignoreSpannableClick;
    }

    /**
     * Call after handling click event for clickable span
     */
    public void preventNextClick() {
        preventClick = true;
    }

    @Override
    public void setOnClickListener(OnClickListener listener) {
        this.clickListener = listener;
        super.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        if (preventClick) {
            preventClick = false;
        } else if (clickListener != null)
            clickListener.onClick(v);
    }
}

由于使用了自定义的LinkMomentMethod,所以((ClickableMovementMethod)getMovementMethod()).onTouchEvent(this, (Spannable)getText(), event);这里做了强转,在Clickable中如下使用:

ClickableSpan is = new ClickableSpan() {
                    @Override
                    public void onClick(View view) {
                        if (view instanceof ClickPreventableTextView){
                            if (((ClickPreventableTextView)view).ignoreSpannableClick())
                                return;
                            //点击ClickableSpan所要执行的操作
                            ((ClickPreventableTextView)view).preventNextClick();
                        }

                    }
                };

ClickPreventableTextView可以照常setOnClickListener设置点击事件,这时我们发现点击ClickableSpan区域并不会触发TextView的onClick事件,注意在ClickableSpanonClick事件执行后,一定要执行((ClickPreventableTextView)view).preventNextClick();

参考链接:
StackOverFlow
陈蒙的博客

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值