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

本文详细介绍了Android中TextView的AutoLink功能及其与ClickSpan长按事件冲突的问题。通过分析`Linkify`和`URLSpan`的工作原理,揭示了autolink的点击事件处理。同时,探讨了TextView的事件分发机制,指出URLSpan的onClick方法在TextView的onTouchEvent中被调用。最后,文章提到了随着短视频流行,NDK开发的重要性日益凸显,尤其是音视频开发的知识需求增加。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


if (mAutoLinkMask != 0) {

Spannable s2;

if (type == BufferType.EDITABLE || text instanceof Spannable) {

s2 = (Spannable) text;

} else {

s2 = mSpannableFactory.newSpannable(text);

}

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

}

}

}

  • 首先调用 Linkify.addLinks 方法解析 autolink 的相关属性

  • 判断是否 mLinksClickable mLinksClickable && !textCanBeSelected() ,若返回 true, 设置 setMovementMethod

我们先来看一下 Linkify 类, 里面定义了几个常量, 分别对应 web , email ,phone ,map,他们的值是位上错开的,这样定义的好处是

  • 方便组合多种值

  • 组合值之后不会丢失状态,即可以获取是否含有某种状态, web, email, phone , map

public class Linkify {

public static final int WEB_URLS = 0x01;

public static final int EMAIL_ADDRESSES = 0x02;

public static final int PHONE_NUMBERS = 0x04;

public static final int MAP_ADDRESSES = 0x08;

}

看一下 linkify 的 addLinks 方法

  • 根据 mask 的标志位,进行相应的正则表达式进行匹配,找到 text 里面的相应的 WEB_URLS, EMAIL_ADDRESSES, PHONE_NUMBERS, MAP_ADDRESSES. 并将相应的文本从 text 里面移除,封装成 LinkSpec,并添加到 links 里面

  • 遍历 links,设置相应的 URLSpan

private static boolean addLinks(@NonNull Spannable text, @LinkifyMask int mask,

@Nullable Context context) {

if (mask == 0) {

return false;

}

URLSpan[] old = text.getSpans(0, text.length(), URLSpan.class);

for (int i = old.length - 1; i >= 0; i–) {

text.removeSpan(old[i]);

}

ArrayList links = new ArrayList();

/ / 根据正则表达式提取 text 里面相应的 WEB_URLS,并且从 text 移除

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

gatherLinks(links, text, Patterns.AUTOLINK_WEB_URL,

new String[] { “http://”, “https://”, “rtsp://” },

sUrlMatchFilter, null);

}

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

gatherLinks(links, text, Patterns.AUTOLINK_EMAIL_ADDRESS,

new String[] { “mailto:” },

null, null);

}

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

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 拦截事件

因此我们猜测 URLS 《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》开源 pan 的 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 (touchI Android开源项目:ali1024.coding.net/public/P7/Android/git sFinished && (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);

}

结语

  • 现在随着短视频,抖音,快手的流行NDK模块开发也显得越发重要,需要这块人才的企业也越来越多,随之学习这块的人也变多了,音视频的开发,往往是比较难的,而这个比较难的技术就是NDK里面的技术。
  • 音视频/高清大图片/人工智能/直播/抖音等等这年与用户最紧密,与我们生活最相关的技术一直都在寻找最终的技术落地平台,以前是windows系统,而现在则是移动系统了,移动系统中又是以Android占比绝大部分为前提,所以AndroidNDK技术已经是我们必备技能了。
  • 要学习好NDK,其中的关于C/C++,jni,Linux基础都是需要学习的,除此之外,音视频的编解码技术,流媒体协议,ffmpeg这些都是音视频开发必备技能,而且
  • OpenCV/OpenGl/这些又是图像处理必备知识,下面这些我都是当年自己搜集的资料和做的一些图,因为当年我就感觉视频这块会是一个大的趋势。所以提前做了一些准备。现在拿出来分享给大家。


vent(this, mSpannable, event);

}

结语

  • 现在随着短视频,抖音,快手的流行NDK模块开发也显得越发重要,需要这块人才的企业也越来越多,随之学习这块的人也变多了,音视频的开发,往往是比较难的,而这个比较难的技术就是NDK里面的技术。
  • 音视频/高清大图片/人工智能/直播/抖音等等这年与用户最紧密,与我们生活最相关的技术一直都在寻找最终的技术落地平台,以前是windows系统,而现在则是移动系统了,移动系统中又是以Android占比绝大部分为前提,所以AndroidNDK技术已经是我们必备技能了。
  • 要学习好NDK,其中的关于C/C++,jni,Linux基础都是需要学习的,除此之外,音视频的编解码技术,流媒体协议,ffmpeg这些都是音视频开发必备技能,而且
  • OpenCV/OpenGl/这些又是图像处理必备知识,下面这些我都是当年自己搜集的资料和做的一些图,因为当年我就感觉视频这块会是一个大的趋势。所以提前做了一些准备。现在拿出来分享给大家。

[外链图片转存中…(img-oUPRA0fG-1649943193170)]

[外链图片转存中…(img-DqB7tblT-1649943193171)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值