需求原因

最近工作中遇到一个需求,后来通过查找相关的资料终于解决了,索性记录下来分享给大家,需要的朋友们可以参考学习。
该需求如下:
**产品说,我们要实现问答功能,答案内的链接要使用内置的浏览器打开。
**视觉说,我们要给超链接标上我们自己的颜色。
如图:
下面我们分析下如何实现。
使用Html
常规方法,给定一段标准html文档,使用Html.fromHtml()封装,直接使用TextView显示。
TextView textView = (TextView) findViewById(R.id.detailed_question_tv_answer); String testString = "亲,一般遇到这问题您可以这样哦:<br>1.可以<font color='#ff8785'><a href='http://m.kaola.com'>催发货</a></font>哦~<br>2.然后耐心等待哦~<br>3.1-3天后新也可以拨打我们的客服."; textView.setMovementMethod(LinkMovementMethod.getInstance()); // 设置链接颜色 textView.setLinkTextColor(getResources().getColor(R.color.red_ff8785)); Spanned htmlString = Html.fromHtml(testString); textView.setText(htmlString);
使用常规方法无论怎么设置,链接都会使用隐式Intent打开,即使用外部的浏览器打开,不符合咱们产品的需求呀。怎么才能监听这个使用并使用内部WebView打开呢?使用SpannableStringBuilder。
使用SpannableStringBuilder
直接上代码。
TextView textView = (TextView) findViewById(R.id.detailed_question_tv_answer);
String testString =
"亲,一般遇到这问题您可以这样哦:<br>1.可以<font color='#ff8785'><a href='http://m.kaola.com'>催发货</a></font>哦~<br>2.然后耐心等待哦~<br>3.1-3天后新也可以拨打我们的客服.";
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setLinkTextColor(getResources().getColor(R.color.red_ff8785));
String linkText = "催发货";
int startIndexOfLink = testString.indexOf(linkText);
int endIndexOfLink = startIndexOfLink + linkText.length();
SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder(testString);
spannableStringBuilder.setSpan(new ClickableSpan() {
@Override
public void onClick(View widget) {
ActivityUtils.startWebviewActivity(DetailedQuestionActivity.this, "http://m.kaola.com", false);
}
}, startIndexOfLink, endIndexOfLink, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableStringBuilder);
当然,这个方法是有很大的局限性的,必须知道链接在文案中的具体位置,以及链接的地址才能够使用这种方法。按照这种思路,我们必须使用正则表达式获取对应的a标签才能得到链接。这种方法拿到的链接在文案中的具体位置是难以把握的,很有可能出错。
Html + SpannableStringBuilder
有没有第三种方法,即能够解析到给定文案中的所有Html标签,又能够使用内置的WebView打开这个链接?从第一种方法中,我们直接使用Html.fromHtml()方法拿到对应的Spanned结果,我们可以从这里入手,看看这个方法是怎么解析html标签的
public static Spanned fromHtml(String source, ImageGetter imageGetter,
TagHandler tagHandler) {
// 使用org.ccil.cowan.tagsoup.Parser作为解析器
Parser parser = new Parser();
try {
parser.setProperty(Parser.schemaProperty, HtmlParser.schema);
} catch (org.xml.sax.SAXNotRecognizedException e) {
// Should not happen.
throw new RuntimeException(e);
} catch (org.xml.sax.SAXNotSupportedException e) {
// Should not happen.
throw new RuntimeException(e);
}
// 使用HtmlToSpannedConverter将Ttml转换成Spanned
HtmlToSpannedConverter converter =
new HtmlToSpannedConverter(source, imageGetter, tagHandler,
parser);
return converter.convert();
}
接下来看一下HtmlToSpannedConverter.convert()这个方法。HtmlToSpannedConverter实现了ContentHandler接口,ContentHandler用于处理Xml文档的解析细节。
public Spanned convert() {
mReader.setContentHandler(this);
try {
mReader.parse(new InputSource(new StringReader(mSource)));
} catch (IOException e) {
// We are reading from a string. There should not be IO problems.
throw new RuntimeException(e);
} catch (SAXException e) {
// TagSoup doesn't throw parse exceptions.
throw new RuntimeException(e);
}
// Fix flags and range for paragraph-type markup.
Object[] obj = mSpannableStringBuilder.getSpans(0, mSpannableStringBuilder.length(), ParagraphStyle.class);
for (int i = 0; i < obj.length; i++) {
int start = mSpannableStringBuilder.getSpanStart(obj[i]);
int end = mSpannableStringBuilder.getSpanEnd(obj[i]);
// If the last line of the range is blank, back off by one.
if (end - 2 >= 0) {
if (mSpannableStringBuilder.charAt(end - 1) == '\n' &&
mSpannableStringBuilder.charAt(end - 2) == '\n') {
end--;
}
}
if (end == start) {
mSpannableStringBuilder.removeSpan(obj[i]);
} else {
mSpannableStringBuilder.setSpan(obj[i], start, end, Spannable.SPAN_PARAGRAPH);
}
}
return mSpannableStringBuilder;
}
我们关心Html是如何被转换成Spanned就够了。在整个解析Html的过程中,是通过SpannableStringBuilder.setSpan(Object what, int start, int end, int flags)这个方法不断进行Html->Spanned转换的。例如,遇到一个a标签,则会通过下边的方法设置一个Span,在SpannabbleStringBuilder内部,Span用一个数组表示,是可以累加的。
// 遇到a标签头
private static void startA(SpannableStringBuilder text, Attributes attributes) {
String href = attributes.getValue("", "href");
int len = text.length();
text.setSpan(new Href(href), len, len, Spannable.SPAN_MARK_MARK);
}
// a标签结束
private static void endA(SpannableStringBuilder text) {
int len = text.length();
Object obj = getLast(text, Href.class);
int where = text.getSpanStart(obj);
text.removeSpan(obj);
if (where != len) {
Href h = (Href) obj;
if (h.mHref != null) {
text.setSpan(new URLSpan(h.mHref), where, len,
Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
}
}
可以看到a标签的转换方法,实际上,a标签最后被转换成了一个URLSpan。
看到这里,思路就来了!实际上,Html.fromHtml()方法最后转换成的对象是一个SpannableStringBuilder,我们可以拿到这个对象的引用,然后获取所有的URLSpan,最后把这些URLSpan全部转换成可以监听的事件不就实现了吗?实际上并没有这么简单,URLSpan是一个类,继承自ClickableSpan,覆盖了其中的onClick(View)方法:
public class URLSpan extends ClickableSpan implements ParcelableSpan {
private final String mURL;
public URLSpan(String url) {
mURL = url;
}
public URLSpan(Parcel src) {
mURL = src.readString();
}
public int getSpanTypeId() {
return TextUtils.URL_SPAN;
}
public int describeContents() {
return 0;
}
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(mURL);
}
public String getURL() {
return mURL;
}
@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());
context.startActivity(intent);
}
}
这里已经默认使用了隐式Intent的方式打开Uri。我们不能直接改变URLSpan类的实现方式,但可以继承这个类并覆盖掉它的onClick(View)方法,或者直接继承ClickableSpan。但是,这样还是会有问题,原先的URLSpan早就在解析的时候存在于SpannableStringBuilder中的,我们需要先移除对应的URLSpan,然后再设置自己实现的新的ClickableSpan就可以了。
具体代码如下:
public static SpannableStringBuilder setTextLinkOpenByWebView(final Context context, String answerString) {
if (!TextUtils.isEmpty(answerString)) {
Spanned htmlString = Html.fromHtml(answerString);
if (htmlString instanceof SpannableStringBuilder) {
SpannableStringBuilder spannableStringBuilder = (SpannableStringBuilder) htmlString;
// 取得与a标签相关的Span
Object[] objs = spannableStringBuilder.getSpans(0, spannableStringBuilder.length(), URLSpan.class);
if (null != objs && objs.length != 0) {
for (Object obj : objs) {
int start = spannableStringBuilder.getSpanStart(obj);
int end = spannableStringBuilder.getSpanEnd(obj);
if (obj instanceof URLSpan) {
//先移除这个Span,再新添加一个自己实现的Span。
URLSpan span = (URLSpan) obj;
final String url = span.getURL();
spannableStringBuilder.removeSpan(obj);
spannableStringBuilder.setSpan(new ClickableSpan() {
@Override
public void onClick(View widget) {
ActivityUtils.startWebviewActivity(context, url, true);
}
}, start, end, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
}
}
}
return spannableStringBuilder;
}
}
return new SpannableStringBuilder(answerString);
}
总结
TextView真的是Android里最强大的组件之一,复杂度很高,源码甚至比Activity还要多。正确的使用TextView具有意想不到的效果~文中为了解决TextView组件中的超链接使用App自带的WebView打开这个问题进行了一次探讨,最终通过hook拿到URLSpan,移除它并实现自己的ClickableSpan,最终解决问题。好了,以上就是这篇文章的全部内容了,希望本文的内容对各位Android开发们能带来一定的帮助,如果有疑问大家可以留言交流。
# webview
# 超链接
# 点击超链接
# textview
# Android 如何使用短信链接打开APP
# Android编程实现点击链接打开APP功能示例
# Android应用中实现跳转外部浏览器打开链接功能
# 转换成
# 自己的
# 是一个
# 移除
# 客服
# 您可以
# 我们可以
# 标上
# 种方法
# 耐心等待
# 的是
# 实现了
# 文档
# 来了
# 会有
# 成了
# 好了
# 这种方法
# 是有
相关文章:
如何用AWS免费套餐快速搭建高效网站?
如何快速搭建高效WAP手机网站?
三星网站视频制作教程下载,三星w23网页如何全屏?
建站ABC备案流程中有哪些关键注意事项?
如何制作网站标识牌,动态网站如何制作(教程)?
php能控制zigbee模块吗_php通过串口与cc2530 zigbee通信【介绍】
昆明网站制作哪家好,昆明公租房申请网上登录入口?
合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?
简易网站制作视频教程,使用记事本编写一个简单的网页html文件?
如何在阿里云购买域名并搭建网站?
实现点击下箭头变上箭头来回切换的两种方法【推荐】
青岛网站建设如何选择本地服务器?
常州企业网站制作公司,全国继续教育网怎么登录?
建站主机空间推荐 高性价比配置与快速部署方案解析
免费视频制作网站,更新又快又好的免费电影网站?
国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?
如何快速生成专业多端适配建站电话?
广东专业制作网站有哪些,广东省能源集团有限公司官网?
高防服务器租用如何选择配置与防御等级?
广德云建站网站建设方案与建站流程优化指南
如何挑选优质建站一级代理提升网站排名?
网站制作公司,橙子建站是合法的吗?
如何在建站之星绑定自定义域名?
零服务器AI建站解决方案:快速部署与云端平台低成本实践
广州商城建站系统开发成本与周期如何控制?
如何通过商城自助建站源码实现零基础高效建站?
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
建站之星后台密码如何安全设置与找回?
建站之星如何助力企业快速打造五合一网站?
外贸公司网站制作,外贸网站建设一般有哪些步骤?
PHP 500报错的快速解决方法
如何规划企业建站流程的关键步骤?
如何在阿里云部署织梦网站?
建站之星价格显示格式升级,你的预算足够吗?
建站DNS解析失败?如何正确配置域名服务器?
微信小程序 五星评分(包括半颗星评分)实例代码
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
,有什么在线背英语单词效率比较高的网站?
建站之星如何快速解决建站难题?
C++如何使用std::optional?(处理可选值)
长沙做网站要多少钱,长沙国安网络怎么样?
网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?
,南京靠谱的征婚网站?
深圳网站制作案例,网页的相关名词有哪些?
外贸公司网站制作哪家好,maersk船公司官网?
南宁网站建设制作定制,南宁网站建设可以定制吗?
清除minerd进程的简单方法
佛山企业网站制作公司有哪些,沟通100网上服务官网?
建站主机是否等同于虚拟主机?
网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?
*请认真填写需求信息,我们会在24小时内与您取得联系。