美文网首页Android
android:使用TextView展示H5文本(含关键字点击和

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

作者: CnPeng | 来源:发表于2017-10-23 16:50 被阅读43次

    一、需求描述

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

    具体如下图:


    使用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;
        }
    }
    

    四、附录

    相关文章

      网友评论

        本文标题:android:使用TextView展示H5文本(含关键字点击和

        本文链接:https://www.haomeiwen.com/subject/wfyauxtx.html