一、需求描述

使用TextView展示H5文本,文本中包含关键词和图片,在H5文本中只有关键词会标红,在TextView中展示出来的关键词需要加点击事件。

具体如下图:



android:使用TextView展示H5文本(含关键字点击和图片)_android



使用TextView展示H5文本



二、需求分析——主要知识点

(1)、使用SpannableString

因H5文本中包含超链接和图片,而且我们要使用TextView展示,那就必须使用SpannableString。将H5文本格式化成Spanned之后再成SpannableString,然后添加 ClickSpan 实现点击事件

(2)、如何解析H5文本获取全部关键词?

需求描述中有说明:在H5文本中只有关键词会标红,所以我们可以根据<font> 节点获取全部的关键词,获取之后存储在set中实现关键词去重。

解析H5文本的时候我们可以自己去解析,也可以直接使用​​ jsoup 库 ​​。


jsoup 是一个开源的H5文本解析库,文中使用的是jsoup库。
添加依赖的时候,直接在 ​​ProjectStructure——Dependences​​中搜索添加即可; 或者直接在 gradle文件中添加​compile 'org.jsoup:jsoup:1.10.3'​


(3)、如何展示图片?

使用 Html.fromHtml(str , ImageGetter , tagHandler) 方法格式化H5 文本时,ImageGetter 可以实现图片的加载。由于图片加载是耗时操作,需要将此代码放置在线程中,防止主线程阻塞。Html.fromHtml(str) 方法不支持图片的展示。

(4)、如何给所有关键词加点击事件?

加点击事件的时候无疑要使用 ClickSpan, 前面我们也已经获取到了全部关键词,而 setClickSpan 的时候需要用到关键词的索引,那么接下来我们就需要遍历获取关键词的索引位置。

遍历某一个关键词的时候,我们会获取到起始索引,根据起始索引又能得到结束索引。获取到该关键词第一次出现时的索引之后,我们需要将字符串进行截取,在截取之后的字符串中继续查找该关键词出现的位置,这样得到的位置是在截取之后的字符串中的位置,我们还要得到关键词在原始字符串中的位置,然后,依次类推,直到返回的索引为-1 —— -1表示后面的文本中没有该关键词了,才去遍历下一个关键词。

截取的目的是为了找出某个关键词所有的出现位置。

所谓原始字符串,这里指的是 Html.fromHtml() 格式化之后构造的SpannableString。

(5)、文本的滚动处理

TextView 本身具有滚动属性,但是在不同的手机上得到的效果不一致,为了方便控制滚动效果,外层使用ScrollView包裹。

三、具体代码实现:

(1)、activity_showh5text.xml

<?xml version="1.0" encoding="utf-8"?>
<layout>
<data>

</data>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">


<TextView
android:id="@+id/tv_showComplexH5Text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>

</ScrollView>

</layout>

(2)、ShowH5TextActivity.java

代码中的H5文本串范例由于在AS中多次格式化,所以出现了很多 +“” ,不影响正常效果。

/**
* 作者:CnPeng
* <p>
* 时间:2017/10/20:下午3:47
* <p>
* 说明:在TextView中展示H5文本,在H5中关键字标红,其他文本不设置字体。在TextView中需要给关键字增加点击事件,同时TextView中还需要展示出H5中指定的图片
* <p>
* ——使用了数据绑定
* ——在解析这个 H5 文本串时使用的是 jsoup 库。
* ——虽然TextView本身具有滚动属性,但是在不同手机上表现不一样:华为Che1-L20上滑动不流畅,魅族m3-note上滑动后会自动回到顶部。所以用ScrollView 包裹
* ——jsoup中没有找到关于根据TAG和节点文本获取属性值的方法,所以无法通过代码去获取font节点中的属性值。(确实有必要的话可以考虑自己解析h5文本)
* ——使用线程是为了保证图片能加载处理,加载图片是耗时操作,不用子线程的话图片可能会加载不出来
*/

public class ShowH5TextActivity extends AppCompatActivity {
String H5String = "<html>\n" + " <head></head>\n" + " <body>\n" + " <p style=\"text-indent: 2em;\"><span " +
"style=\"font-family: 宋体, SimSun; font-size: 16px;\">周三下午公布的<font " +
"color=\"#FF0000\">英国</font>5月失业率、英国5月失业金申请人数、英国4月三个月ILO失业率显示,英国4月三个月剔除红利的平均工资年率刷新2015年1月以来新低。英国2-4月连续3"
+ "个月失业率为1975年以来最低,英国就业市场连续3" +
"个月保持稳健,但薪资增速进一步放缓,料将对内需产生负面影响,为英国经济增长预期增添担忧情绪。英国国家统计局表示薪资数据将改善小型企业的薪资策略,对薪资水平产生下行影响。</span></p>\n" + " " +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "<p><br " + "/></p>\n" + " " + "<p" + "" + " " +
"style=\"text-indent: " + "2em;" + "\"><span " + "" + "" + "" + "style=\"font-family:" + " " + "宋体, " +
"" + "SimSun; " + "font-size: " + "16px;" + "\">英国前首相<font " +
"color=\"#FF000\">卡梅伦</font>表示现任首相特雷莎&middot;" +
"梅应采取“软脱欧”,并表示她应该与工党等反对派进行进一步交涉,与各党派进行更广泛的磋商以达成更多共识。认为“软脱欧”或许会面临更大压力,并表示议会现在应尽快面对这个问题。同时<font " +
"color=\"#FF000\">卡梅伦</font>还对特蕾莎&middot;梅表示了支持。</span></p>\n" + " <p><br /></p>\n" + " <p " +
"style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;" +
"\">据华尔街日报,MacroPolicy" + " Perspectives " + "LLC调查显示约50%的受访者表示,股市没有对美联储的计划做出反应,42" +
"%的受访者认为信用债市场也没有做出反应。几乎没有受访者认为美联储的计划在任何市场得到了充分的消化。这表明如果美联储在启动这项计划前没有与市场有效沟通,将可能引发不利的市场变动。</span></p>\n" +
" <p><br /></p>\n" + " <p style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; " +
"font-size:" + " 16px;\">北京时间本周四凌晨2点美联储将公布最新利率决议及<font " +
"color=\"#FF0000\">政策声明</font>,预计美元在经历会议后将面临走软风险;市场广泛预期本次会议将加息25个基点至1.00%-1.25%;然而,FOMC有可能在此次声明中降低核心PCE"
+ "通胀预期,长期联邦利率中值预期也有降低的可能性,这将对加息造成压力;此外,预计本次会议将对缩减资产负债表计划有所置评。</span></p>\n" + " <p><br /></p>\n" + " "
+ "<p " + "style=\"text-indent: 2em;\"><img src=\"http://www.gfxa" + "" + "" + "" + "" + "" + "" + "" +
"" + ".com/upload/image/20170614/6363305821523119573565195.png\" title=\"\" /></p>\n" + " <p " +
"style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;" +
"\">支撑:1260——1255——1247 阻力:1273——1281</span></p>\n" + " <p style=\"text-indent: 2em;\"><span " +
"style=\"font-family: 宋体, SimSun; font-size: 16px;\">交易策略:现货黄金现价1268.30,日内交易建议如下:</span></p>\n" + " <p "
+ "style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;\">A:北京时间22:00之前,"
+ "现货黄金上行至1274附近时四十分之一仓位做空,止损设1279,目标下看至1268/1265区间止盈。持仓阶段,现货黄金下破1271后,建议将止损位下移至1274附近。持仓阶段,浮盈大于6" +
"美金时建议随机止盈。鉴于美联储利率决议影响的不确定性,此交易如触发,北京时间6月15日01:00之前建议择机离场。</span></p>\n" + " <p style=\"text-indent: " +
"2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;\">B:北京时间6月15日07:00之前," +
"现货黄金下行至1260和1253附近时分别以五十分之一仓位做多,止损统一设1245,目标依次上看1268—1273—1281—1288附近。持仓阶段,浮盈大于5" +
"美金时,建议将止损位上移至成本位。持仓阶段,浮盈大于25美金时建议随机止盈。</span></p>\n" + " <p><br /></p>\n" + " <p style=\"text-indent: "
+ "2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;\">美元兑日元</span></p>\n" + " <p " +
"style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;" +
"\">美元兑日元</span></p>\n" + " <p style=\"text-indent: 2em;\"><img src=\"http://www.gfxa" + "" + "" + "" +
".com/upload/image/20170614/6363305822435631122386267.png\" title=\"\" /></p>\n" + " <p " +
"style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun; font-size: 16px;\">支撑109.90 " +
"阻力110.40-110.80</span></p>\n" + " <p style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, SimSun;" +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + " " + "font-size: " + "16px;" +
"\">交易策略:美元兑日元,现价报111.20。明日凌晨有美联储议息会议,注意风险。加息概率极高,但是美元走势依旧疲软,不排除出现美元空头回补的现象。日内交易建议如下:</span" + "></p" +
"" + ">\n" + "" + " " + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" +
"<p " + "style=\"text-indent: " + "2em;" + "\"><span " + "style=\"font-family: " + "宋体, " + "SimSun;" +
"" + " " + "font-size: " + "16px;" + "\">A: " + "突破110.40做多,止损110.30,止盈110.77</span></p>\n" + " " + "<p " +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "style=\"text-indent: 2em;" +
"\"><span " + "" + "style=\"font-family:" + " 宋体, " + "" + "SimSun; " + "font-size: " + "16px;" + "\">B: " +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" +
"" + "应对加息:限价卖出挂单于110.80,止损111.20,止盈110.40</span></p>\n" + "" + " " + "<p><br " + "/></p>\n" + " " + " "
+ "<p " + "style=\"text-indent: " + "2em;\"><span " + "style=\"font-family: 宋体, " + "SimSun; " +
"font-size: 16px;" + "\">英镑兑美元</span></p>\n" + " <p " + "style=\"text-indent: 2em;" + "\"><img" + " " +
"src=\"http://www.gfxa" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" +
"" + "" + "" + "" + "" + "" + "" + "" + ".com/upload/image/20170614/6363305823223140971492124.png\" " +
"title=\"\" " + "/><span" + " " + "style=\"font-family: 宋体, " + "SimSun; font-size: 16px;\">&nbsp; &nbsp;" +
"" + "" + "" + "" + "" + "" + "" + "" + " &nbsp; &nbsp;" + " " + "&nbsp; &nbsp;" + " " + "" + "" + "" +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "&nbsp; &nbsp; &nbsp; &nbsp; " + "&nbsp;" + "" +
" " + "&nbsp; " + "&nbsp; " + "" + "" + "&nbsp; " + "&nbsp; " + "&nbsp;" + " " + "&nbsp; " + "&nbsp; " +
"" + "&nbsp;" + "</span></p>\n" + "" + "" + " <p " + "" + "style=\"text-indent: 2em;" + "\"><span" + " "
+ "style=\"font-family: " + "宋体, " + "" + "SimSun; " + "font-size: " + "16px;" + "\">支撑1.2673-1.2600 " +
"阻力1.1287-1.2828</span></p>\n" + " " + "<p" + " " + "style=\"text-indent:" + " " + "" + "" + "" + "" +
"" + "" + "" + "" + "" + "" + "" + "" + "" + "2em;" + "\"><span " + "" + "" + "style=\"font-family: " +
"宋体, " + "SimSun;" + " " + "font-size: " + "" + "16px;" +
"\">交易策略:如图欧元兑美元四小时图所示,现价报1.2753,欧元轴心点为1.2714,中枢区间为1.2698—1.2730" + ",日内交易建议如下:</span></p>\n" + " " +
"<p " + "style=\"text-indent: 2em;\"><span style=\"font-family: 宋体, " + "SimSun; font-size: " + "16px;" +
"\">A:建议1.2730卖出英镑对美元,止损1.2787,止盈1.2673.</span></p>\n" + " <p " + "style=\"text-indent: 2em;" +
"\"><span " + "style=\"font-family: 宋体, SimSun; font-size: 16px;" +
"\">(该建议以10000美金下0.5手为基准,参照可自行换算。请投资者控制好仓位,严格止损。)</span></p>\n" + " <p><br /></p>\n" + " </body>\n" +
"</html>";

private ActivityShowh5textBinding binding;
private String tempSplitedStr; //临时切割得到的字符串

private final int FLAG_CONVERT_H5TEXT_OVER = 1; //将H5转换成spanableString 完毕
private final int MODE_INTRINSIC = 0x001; //根据图片的原始大小进行展示
private final int MODE_BASE_WINDOW_WITH = 0x002; //与屏幕等宽(保持宽高比)

Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
if (null != msg && msg.what == FLAG_CONVERT_H5TEXT_OVER) {
binding.tvShowComplexH5Text.setText((SpannableString) msg.obj);
//设置该句使文本的超连接起作用,不设置该句代码,点击事件不生效!!!
binding.tvShowComplexH5Text.setMovementMethod(LinkMovementMethod.getInstance());
}
}
};


@Override
public void onCreate(
@Nullable
Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = DataBindingUtil.setContentView(this, R.layout.activity_showh5text);
getAndSetStrToTextView();
}

/**
* 获取并设置字符串到TextView
*/
private void getAndSetStrToTextView() {
final HashSet<String> keyWordsSet = getAllKeyWords();

new Thread(new Runnable() { //之所以放在线程中完成H5转 SpannableStirng ,是为了加载H5的图片
@Override
public void run() {
Spanned normalStr = convertH5TextToSpanned();
SpannableString spannableStr = new SpannableString(normalStr); //最终要展示的字符串

tempSplitedStr = spannableStr.toString(); //全局变量,赋初值

for (String keyStr : keyWordsSet) { //为所有关键字增加点击事件
findKeyAndSetEvent(spannableStr, tempSplitedStr, keyStr, 0);
}

Message msg = handler.obtainMessage();
msg.what = FLAG_CONVERT_H5TEXT_OVER;
msg.obj = spannableStr;
handler.sendMessage(msg);
}
}).start();
}


/**
* 将H5字符串转换成Spanned字符串保证图片的显示。
*/
private Spanned convertH5TextToSpanned() {
return Html.fromHtml(H5String, new Html.ImageGetter() {
@Override
public Drawable getDrawable(String url) {
InputStream is;
try {
is = (InputStream) new URL(url).getContent();
Drawable d = Drawable.createFromStream(is, "src");

setDrawableBounds(d, MODE_BASE_WINDOW_WITH); //设置图片区域

is.close();
return d;
} catch (Exception e) {
return null;
}
}
}, null);
}

/**
* 设置图片的区域,必须设置,否则图片不展示
*
* @param d 图片对象
* @param withOrHeightMode 宽高模式
*/
private void setDrawableBounds(Drawable d, int withOrHeightMode) {
switch (withOrHeightMode) {
case MODE_INTRINSIC: //根据原图大小进行展示
d.setBounds(0, 0, d.getIntrinsicWidth(), d.getIntrinsicHeight());
break;
case MODE_BASE_WINDOW_WITH: //与屏幕等宽
WindowManager wm = getWindowManager();
int wmWidth = wm.getDefaultDisplay().getWidth();
int picWidth = d.getIntrinsicWidth();
int picHeight = d.getIntrinsicHeight();

picHeight = (int) (picHeight * (wmWidth / picWidth * 1.0));

d.setBounds(0, 0, wmWidth, picHeight);
}
}

/**
* 找出单个关键字每一次出现的位置并为其增加点击事件
*
* @param tempSplitedStr 被切割后的新字符串
* @param keyStr 关键字
* @param preEndIndex 关键词上一次出现时的结束索引/关键字本次在原始字符串中的结束索引
*/
private void findKeyAndSetEvent(SpannableString spannableString, String tempSplitedStr, final String keyStr,
int preEndIndex) {
final int startIndex = tempSplitedStr.indexOf(keyStr); //起始索引
if (startIndex != -1) {
final int endIndex = startIndex + keyStr.length() - 1; //终止索引,
int startIndexInOgirinal = 0;

if (preEndIndex == 0) { //关键字第一次出现
startIndexInOgirinal = startIndex;
preEndIndex = endIndex;
} else { //关键字不是第一次出现
startIndexInOgirinal = startIndex + preEndIndex + 1; //加1 是因为截取的字符串索引又是从0开始
preEndIndex = startIndexInOgirinal + keyStr.length() - 1; //减1 是因为起始索引已经占了一个索引
}

LogUtils.e("在临时字符串中的位置:", startIndex + "/" + endIndex);
LogUtils.e("原始字符串中的位置:", startIndexInOgirinal + "/" + preEndIndex);

spannableString.setSpan(new ClickableSpan() {
@Override
public void onClick(View widget) {
//点击事件弹窗+请求服务器数据
Toast.makeText(ShowH5TextActivity.this, "点我干嘛?关键字:" + keyStr, Toast.LENGTH_SHORT).show();
}

@Override
public void updateDrawState(TextPaint ds) {
//super.updateDrawState(ds);
ds.setColor(Color.RED); //更改超链接颜色(此颜色要与H5中关键字的 font 颜色一致)
ds.setUnderlineText(false); //不展示下划线
}
}, startIndexInOgirinal, preEndIndex + 1, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
//}, startIndexInOgirinal, preEndIndex, Spanned.SPAN_INCLUSIVE_INCLUSIVE); //这样的话,非第一次出现的只会将第一个字符加上超链接

tempSplitedStr = tempSplitedStr.substring(endIndex + 1); //截取字符串,+1 表示从关键词后面截取,不含关键字;不加1 的话从关键词最后一个字开始截取
findKeyAndSetEvent(spannableString, tempSplitedStr, keyStr, preEndIndex); //递归调用
}
}

/**
* 获取关键字,并使用Set存储,实现去重
*/
private HashSet<String> getAllKeyWords() {
HashSet<String> keysSet = new HashSet<>();
Document document = Jsoup.parse(H5String);
Elements elementsList = document.getElementsByTag("font"); //在JSOUP中,Elements类继承自ArrayList

if (null != elementsList) {
for (Element element : elementsList) {
keysSet.add(element.text());
}
}

return keysSet;
}
}

四、附录