TextView AutoLink, ClikSpan 与长按事件冲突的解决(1)

gatherTelLinks(links, text, context);

}

if ((mask & MAP_ADDRESSES) != 0) {

gatherMapLinks(links, text);

}

pruneOverlaps(links);

if (links.size() == 0) {

return false;

}

// 遍历 links,设置相应的 URLSpan

for (LinkSpec link: links) {

applyLink(link.url, link.start, link.end, text);

}

return true;

}

private static final void applyLink(String url, int start, int end, Spannable text) {

URLSpan span = new URLSpan(url);

text.setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

}

接下来我们一起来看一下这个 URLSpan 是何方神圣,它继承了 ClickableSpan(注意下文会用到它),并且重写了 onClick 方法,我们可以看到在 onClick 方法里面,他通过相应的 intent 取启动相应的 activity。因此,我们可以断定 autolink 的自动跳转是在这里处理的。

public class URLSpan extends ClickableSpan implements ParcelableSpan {

private final String mURL;

/**

  • Constructs a {@link URLSpan} from a url string.

  • @param url the url string

*/

public URLSpan(String url) {

mURL = url;

}

/**

  • Constructs a {@link URLSpan} from a parcel.

*/

public URLSpan(@NonNull Parcel src) {

mURL = src.readString();

}

@Override

public int getSpanTypeId() {

return getSpanTypeIdInternal();

}


@Override

public void onClick(View widget) {

Uri uri = Uri.parse(getURL());

Context context = widget.getContext();

Intent intent = new Intent(Intent.ACTION_VIEW, uri);

intent.putExtra(Browser.EXTRA_APPLICATION_ID, context.getPackageName());

try {

context.startActivity(intent);

} catch (ActivityNotFoundException e) {

Log.w(“URLSpan”, "Actvity was not found for intent, " + intent.toString());

}

}

}

解决了 autolink 属性点击事件在哪里响应了,接下来我们一起看一下 URLSpan 的 onClick 方法是在哪里调用的。

autolink 的 onclick 事件是在哪里被调用的

我们先来复习一下 View 的事件分发机制:

  • dispatchTouchEvent ,这个方法主要是用来分发事件的

  • onInterceptTouchEvent,这个方法主要是用来拦截事件的(需要注意的是ViewGroup才有这个方法,- View没有onInterceptTouchEvent这个方法

  • onTouchEvent 这个方法主要是用来处理事件的

requestDisallowInterceptTouchEvent(true),这个方法能够影响父View是否拦截事件,true 表示父 View 不拦截事件,false 表示父 View 拦截事件

因此我们猜测 URLSpan 的 onClick 事件是在 TextView 的 onTouchEvent 事件里面调用的。下面让我们一起来看一下 TextView 的 onTouchEvent 方法

@Override

public boolean onTouchEvent(MotionEvent event) {

final int action = event.getActionMasked();

if (mEditor != null) {

mEditor.onTouchEvent(event);

if (mEditor.mSelectionModifierCursorController != null

&& mEditor.mSelectionModifierCursorController.isDragAcceleratorActive()) {

return true;

}

}

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 the long press should have triggered an

  • insertion action mode, we can now actually show it.

*/

if (mEditor != null && mEditor.mDiscardNextActionUp && action == MotionEvent.ACTION_UP) {

mEditor.mDiscardNextActionUp = false;

if (mEditor.mIsInsertionActionModeStartPending) {

mEditor.startInsertionActionMode();

mEditor.mIsInsertionActionModeStartPending = 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, mSpannable, 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 = mSpannable.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 (isTextEditable() && mEditor.mShowSoftInputOnFocus && 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;

}

首先如果 mEditor != null 会将touch事件交给mEditor处理,这个 mEditor 其实是和 EditText 有关系的,没有使用 EditText 这里应该是不会被创建的。

去除 mEditor != null 的相关逻辑之后,剩下的相关代码主要如下:

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, mSpannable, 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 = mSpannable.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 (isTextEditable() && mEditor.mShowSoftInputOnFocus && 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;

}

}

首先我们先来看一下, mMovement 是否可能为 null,若不为 null,则会调用 handled |= mMovement.onTouchEvent(this, mSpannable, event) 方法。

找啊找,发现在 setText 里面有调用这一段代码,setMovementMethod(LinkMovementMethod.getInstance()); 即 mLinksClickable && !textCanBeSelected() 为 true 的时候给 TextView 设置 MovementMethod。

查看 TextView 的源码我们容易得知 mLinksClickable 的值默认为 true, 而 textCanBeSelected 方法会返回 false,即 mLinksClickable && !textCanBeSelected() 为 true,这个时候会给 TextView 设置 setMovementMethod。 因此在 TextView 的 onTouchEvent 方法中,若 autoLink 等于 true,并且 text 含有 email,phone, webAddress 等的时候,会调用 mMovement.onTouchEvent(this, mSpannable, event) 方法。

if (Linkify.addLinks(s2, mAutoLinkMask)) {

text = s2;

type = (type == BufferType.EDITABLE) ? BufferType.EDITABLE : BufferType.SPANNABLE;

/*

  • We must go ahead and set the text before changing the

  • movement method, because setMovementMethod() may call

  • setText() again to try to upgrade the buffer type.

*/

setTextInternal(text);

// Do not change the movement method for text that support text selection as it

// would prevent an arbitrary cursor displacement.

if (mLinksClickable && !textCanBeSelected()) {

setMovementMethod(LinkMovementMethod.getInstance());

}

}

boolean textCanBeSelected() {

// prepareCursorController() relies on this method.

// If you change this condition, make sure prepareCursorController is called anywhere

// the value of this condition might be changed.

// 默认 mMovement 为 null

if (mMovement == null || !mMovement.canSelectArbitrarily()) return false;

return isTextEditable()

|| (isTextSelectable() && mText instanceof Spannable && isEnabled());

}

ok ,我们一起在来看一下 mMovement 的 onTouchEvent 方法

MovementMethod 是一个借口,实现子类有 ArrowKeyMovementMethod, LinkMovementMethod, ScrollingMovementMethod 。

这里我们先来看一下 LinkMovementMethod 的 onTouchEvent 方法

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[] links = buffer.getSpans(off, off, ClickableSpan.class);

if (links.length != 0) {

ClickableSpan link = links[0];

if (action == MotionEvent.ACTION_UP) {

if (link instanceof TextLinkSpan) {

((TextLinkSpan) link).onClick(

widget, TextLinkSpan.INVOCATION_METHOD_TOUCH);

} else {

link.onClick(widget);

}


}

这里我们重点关注代码 20 - 31 行,可以看到,他会先取出所有的 ClickableSpan,而我们的 URLSpan 正是 ClickableSpan 的子类,接着判断是否是 ACTION_UP 事件,然后调用 onClick 事件。因此,ClickableSpan 的 onClick 方法是在 ACTION_UP 事件中调用的,跟我们的长按事件没半毛钱关系。

重要的事情说三遍

ClickableSpan 的 onClick 方法是在 ACTION_UP 事件中调用的

ClickableSpan 的 onClick 方法是在 ACTION_UP 事件中调用的

ClickableSpan 的 onClick 方法是在 ACTION_UP 事件中调用的

知道了 ClickableSpan 的 onClick 方法是在 ACTION_UP 事件中调用的,下面让我们一起来看一下怎样解决 TextView 中 autolink 与 clickableSpan 与长按事件的冲突。


解决思路


其实很简单,既然,它是在 ACTION_UP 事件处理的,那么我们只需要监听到长按事件,并且当前 MotionEvent 是 ACTION_UP 的时候,我们直接返回 true,不让他继续往下处理就 ok 了。

由于时间关系,没有详细去了解 View 的长按事件的促发事件,这里我们已按下的事件超过 500 s,即使别为长按事件。

这里,我们定义一个 ControlClickSpanTextView,继承 AppCompatTextView,代码如下。

  • 在 ACTION_DOWN 的时候记录下事件

  • ACTION_UP 的时候,判断事件是否超过 500 毫秒,超过 500 毫秒,不再处理事件,直接返回 true

public class ControlClickSpanTextView extends AppCompatTextView {

private static final String TAG = “AutoLinkTextView”;

private long mTime;

private boolean mLinkIsResponseLongClick = false;

public ControlClickSpanTextView(Context context) {

super(context);

}

public ControlClickSpanTextView(Context context, AttributeSet attrs) {

super(context, attrs);

}

public ControlClickSpanTextView(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

}

public boolean isLinkIsResponseLongClick() {

return mLinkIsResponseLongClick;

}

public void setLinkIsResponseLongClick(boolean linkIsResponseLongClick) {

this.mLinkIsResponseLongClick = linkIsResponseLongClick;

}

@Override

public boolean onTouchEvent(MotionEvent event) {

CharSequence text = getText();

if (text == null) {

return super.onTouchEvent(event);

}

if (!mLinkIsResponseLongClick && text instanceof Spannable) {

int end = text.length();

Spannable spannable = (Spannable) text;

ClickableSpan[] clickableSpans = spannable.getSpans(0, end, ClickableSpan.class);

if (clickableSpans == null || clickableSpans.length == 0) {

return super.onTouchEvent(event);

}

最后

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

上面分享的腾讯、头条、阿里、美团、字节跳动等公司2019-2021年的高频面试题,博主还把这些技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,上面只是以图片的形式给大家展示一部分。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

【Android高级架构视频学习资源】
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
instanceof Spannable) {

int end = text.length();

Spannable spannable = (Spannable) text;

ClickableSpan[] clickableSpans = spannable.getSpans(0, end, ClickableSpan.class);

if (clickableSpans == null || clickableSpans.length == 0) {

return super.onTouchEvent(event);

}

最后

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

上面分享的腾讯、头条、阿里、美团、字节跳动等公司2019-2021年的高频面试题,博主还把这些技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,上面只是以图片的形式给大家展示一部分。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

[外链图片转存中…(img-WrZOCPJT-1715876346840)]

【Android高级架构视频学习资源】
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 5
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值