Android textview展示富文本内容

image.png

今天实现的内容,就是上图的效果,通过Span方式展示图片,需要支持文字颜色改变、加粗。支持style=\"color:green; font-weight:bold;\"展示。尤其style标签中的font-sizefont-weight是在原生中不被支持的。

所以我们今天需要使用自定义的方式来实现实现。

val result = "<spanExt>攀钢钒钛所属行业为\n" +  
"<spanExt style=\"color:#333333; font-weight:bold; font-size:18px;\">其他采掘</spanExt>;\n" +  
"</spanExt>\n" +  
"<img src=\"https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png\">" +  
"<br/>\n" +  
"<spanExt>当日攀钢钒钛行情整体表现<spanExt style=\"color:green; font-weight:bold;\">弱于</spanExt>所属行业表现;</spanExt>\n" +  
"<br/>\n" +  
"<img src=\"https://hbimg.huaban.com/4829401262ba0574fe8328b9f7f4b871d53850df7cd4-wVjrLB_fw658\">" +  
"<spanExt>攀钢钒钛所属概念中<spanExt style=\"color:#333333; font-weight:bold; \">有色金属</spanExt>表现相对优异;</spanExt>\n" +  
"<br/>\n" +  
"<img src=\"https://bkimg.cdn.bcebos.com/pic/50da81cb39dbb6fd526675ca147cbc18972bd507999d?x-bce-process=image/watermark,image_d2F0ZXIvYmFpa2U5Mg==,g_7,xp_5,yp_5/format,f_auto\">" +  
"<spanExt>其涨跌幅在有色金属中位列<spanExt style=\"color:#F43737; font-weight:bold; \">81</spanExt>/<spanExt style=\"color:black; font-weight:bold; \">122</spanExt></spanExt>"

计划渲染的目标内容,这里的spanExt是由于系统本身是支持span标签,但是对于span标签的支持不够完善,我需要自定义一个新的标签,但是html解析的时候,是识别的标签,所以需要将展示内容的span标签替换为了spanExt。防止被系统的span标签直接解析了。

渲染内容中,主要是需要自定义span标签和图片的展示。接下来就从这两个方面出发说明。

自定义tag优化span标签

package org.fireking.basic.textview.html;import android.content.Context;
import android.content.res.ColorStateList;
import android.content.res.Resources;
import android.graphics.Color;
import android.graphics.Typeface;
import android.text.Editable;
import android.text.Html;
import android.text.Spannable;
import android.text.Spanned;
import android.text.TextUtils;
import android.text.style.AbsoluteSizeSpan;
import android.text.style.ForegroundColorSpan;
import android.text.style.StyleSpan;
import android.text.style.TextAppearanceSpan;
import android.util.Log;import org.xml.sax.XMLReader;import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.Map;public class SpanExtTagHandler implements Html.TagHandler {private final String TAG = "CustomTagHandler";private int startIndex = 0;private int stopIndex = 0;private final ColorStateList mOriginColors;private final Context mContext;public SpanExtTagHandler(Context context, ColorStateList originColors) {mContext = context;mOriginColors = originColors;}@Overridepublic void handleTag(boolean opening, String tag, Editable output,XMLReader xmlReader) {processAttributes(xmlReader);if (tag.equalsIgnoreCase("spanExt")) {if (opening) {startSpan(tag, output, xmlReader);} else {endSpan(tag, output, xmlReader);attributes.clear();}}}public void startSpan(String tag, Editable output, XMLReader xmlReader) {startIndex = output.length();}public void endSpan(String tag, Editable output, XMLReader xmlReader) {stopIndex = output.length();String color = attributes.get("color");String size = attributes.get("size");String style = attributes.get("style");if (!TextUtils.isEmpty(style)) {analysisStyle(startIndex, stopIndex, output, style);}if (!TextUtils.isEmpty(size)) {size = size.split("px")[0];}if (!TextUtils.isEmpty(color)) {if (color.startsWith("@")) {Resources res = Resources.getSystem();String name = color.substring(1);int colorRes = res.getIdentifier(name, "color", "android");if (colorRes != 0) {output.setSpan(new ForegroundColorSpan(colorRes), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}} else {try {output.setSpan(new ForegroundColorSpan(Color.parseColor(color)), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);} catch (Exception e) {e.printStackTrace();reductionFontColor(startIndex, stopIndex, output);}}}if (!TextUtils.isEmpty(size)) {int fontSizePx = 16;if (null != mContext) {fontSizePx = DisplayUtil.sp2px(mContext, Integer.parseInt(size));}output.setSpan(new AbsoluteSizeSpan(fontSizePx), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}}final HashMap<String, String> attributes = new HashMap<String, String>();private void processAttributes(final XMLReader xmlReader) {try {Field elementField = xmlReader.getClass().getDeclaredField("theNewElement");elementField.setAccessible(true);Object element = elementField.get(xmlReader);Field attsField = element.getClass().getDeclaredField("theAtts");attsField.setAccessible(true);Object atts = attsField.get(element);Field dataField = atts.getClass().getDeclaredField("data");dataField.setAccessible(true);String[] data = (String[]) dataField.get(atts);Field lengthField = atts.getClass().getDeclaredField("length");lengthField.setAccessible(true);int len = (Integer) lengthField.get(atts);for (int i = 0; i < len; i++)attributes.put(data[i * 5 + 1], data[i * 5 + 4]);} catch (Exception e) {e.printStackTrace();}}/*** 还原为原来的颜色*/private void reductionFontColor(int startIndex, int stopIndex, Editable editable) {if (null != mOriginColors) {editable.setSpan(new TextAppearanceSpan(null, 0, 0, mOriginColors, null),startIndex, stopIndex,Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);} else {editable.setSpan(new ForegroundColorSpan(0xff2b2b2b), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}}/*** 解析style属性*/private void analysisStyle(int startIndex, int stopIndex, Editable editable, String style) {Log.e(TAG, "style:" + style);String[] attrArray = style.split(";");Map<String, String> attrMap = new HashMap<>();if (null != attrArray) {for (String attr : attrArray) {String[] keyValueArray = attr.split(":");if (null != keyValueArray && keyValueArray.length == 2) {// 记住要去除前后空格attrMap.put(keyValueArray[0].trim(), keyValueArray[1].trim());}}}Log.e(TAG, "attrMap:" + attrMap.toString());String color = attrMap.get("color");String fontSize = attrMap.get("font-size");String fontWeight = attrMap.get("font-weight");if (!TextUtils.isEmpty(fontWeight) && "bold".equalsIgnoreCase(fontWeight)) {editable.setSpan(new StyleSpan(Typeface.BOLD), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);} else if (!TextUtils.isEmpty(fontWeight) && "italic".equalsIgnoreCase(fontWeight)) {editable.setSpan(new StyleSpan(Typeface.ITALIC), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}if (!TextUtils.isEmpty(fontSize)) {fontSize = fontSize.split("px")[0];}if (!TextUtils.isEmpty(color)) {if (color.startsWith("@")) {Resources res = Resources.getSystem();String name = color.substring(1);int colorRes = res.getIdentifier(name, "color", "android");if (colorRes != 0) {editable.setSpan(new ForegroundColorSpan(colorRes), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}} else {try {editable.setSpan(new ForegroundColorSpan(Color.parseColor(color)), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);} catch (Exception e) {e.printStackTrace();reductionFontColor(startIndex, stopIndex, editable);}}}if (!TextUtils.isEmpty(fontSize)) {int fontSizePx = 16;if (null != mContext) {fontSizePx = DisplayUtil.sp2px(mContext, Integer.parseInt(fontSize));}editable.setSpan(new AbsoluteSizeSpan(fontSizePx), startIndex, stopIndex, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);}}
}  

代码整体上非常简单,首先判断需要解析的标签。根据#public void handleTag(boolean opening, String tag, Editable output, XMLReader xmlReader)的isOpen判断是开标签还是关标签。

> 开标签<span 
> 关标签</span>

对于开标签,不需要过多关注,只需要知道开始的位置即可。重点关注的还是关标签。

这里需要使用xmlReader解析出来对应的属性和文本内容,然后将内容转换为对应的span设置给Editable output即可。

自定义Html.ImageGetter支持加载网络图片

  • BitmapTarget.java

BitmapTarget用户装载Glide加载的图片对象

public class BitmapTarget extends SimpleTarget<Bitmap> {  private final DrawableWrapper drawableWrapper;  private Context context;  private TextView textView;  public BitmapTarget(TextView textView, DrawableWrapper drawableWrapper, Context context) {  this.drawableWrapper = drawableWrapper;  this.context = context;  this.textView = textView;  }  @Override  public void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {  Drawable drawable = new BitmapDrawable(context.getResources(), resource);  int width = drawable.getIntrinsicWidth();  int height = drawable.getIntrinsicHeight();  drawable.setBounds(0, 0, width, height);  drawableWrapper.setBounds(0, 0, width, height);  drawableWrapper.setDrawable(drawable);  textView.setText(textView.getText());  textView.invalidate();  }  
}
  • DrawableWrapper.java

DrawableWrapper用来承接渲染到ImageSpandrawable

public class DrawableWrapper extends BitmapDrawable {  private Drawable drawable;  DrawableWrapper() {  }  @Override  public void draw(Canvas canvas) {  if (drawable != null){  drawable.draw(canvas);  }  }  public Drawable getDrawable() {  return drawable;  }  public void setDrawable(Drawable drawable) {  this.drawable = drawable;  }  
}
  • MyImageGetter.java

MyImageGetter 使用Glide进行图片的下载,并且渲染到drawable中,用于ImageSpan展示使用。

public class MyImageGetter implements Html.ImageGetter {  private Context context;  private TextView textView;  public MyImageGetter(Context context, TextView textView) {  this.context = context;  this.textView = textView;  }  @Override  public Drawable getDrawable(String source) {  DrawableWrapper drawableWrapper = new DrawableWrapper();  Drawable drawable = new BitmapDrawable(Bitmap.createBitmap(200, 200, Bitmap.Config.ARGB_8888));  drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());  drawableWrapper.setDrawable(drawable);  Glide.with(context).asBitmap().load(source).into(new BitmapTarget(textView, drawableWrapper, context));  return drawableWrapper;  }  
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/315966.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

竞赛保研 基于机器视觉的二维码识别检测 - opencv 二维码 识别检测 机器视觉

文章目录 0 简介1 二维码检测2 算法实现流程3 特征提取4 特征分类5 后处理6 代码实现5 最后 0 简介 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于机器学习的二维码识别检测 - opencv 二维码 识别检测 机器视觉 该项目较为新颖&#xff0c;适合作为竞赛课…

【docker】如何编写dockerfile文件,构建docker镜像

如何编写dockerfile文件&#xff0c;构建docker镜像 一、docker 镜像与 dockerfile1.1 什么是Docker镜像1.2 Docker 镜像的结构 二、dockerfile 中常用的构建指令三、dockerfile 内容示例四、构建 docker 镜像 一、docker 镜像与 dockerfile 1.1 什么是Docker镜像 Docker镜像…

目标检测-Owo Stage-YOLOv2

文章目录 前言一、YOLOv2的网络结构和流程二、YOLOv2的创新点预处理网络结构训练 总结 前言 根据前文目标检测-One Stage-YOLOv1可以看出YOLOv1的主要缺点是&#xff1a; 和Fast-CNN相比&#xff0c;速度快&#xff0c;但精度下降。&#xff08;边框回归不加限制&#xff09;…

Eureka服务注册与发现中心

简介 Spring Cloud封装了Netflix 公司开发的Eureka模块来实现服务治理 在传统的RPC远程调用框架中&#xff0c;管理每个服务与服务之间依赖关系比较复杂&#xff0c;管理比较复杂&#xff0c;所以需要使用服务治理&#xff0c;管理服务于服务之间依赖关系&#xff0c;可以实现…

自动化测试之等待方式详解

在自动化测试中&#xff0c;等待是一个重要的技术&#xff0c;用于处理页面加载、元素定位、元素状态改变等延迟问题。 等待能够确保在条件满足后再进行后续操作&#xff0c;提高自动化测试的稳定性以及可靠性。 等待方式&#xff1a;显示等待、隐式等待、线程睡眠 1. 显式等…

TCP_可靠数据传输原理

引言 在网络通信中&#xff0c;TCP是确保数据可靠传输的关键协议。但在我们深入研究TCP拥塞控制技术之前&#xff0c;让我们先探索可靠数据传输的原理&#xff0c;特别是TCP头部中一些重要字段的作用。 网络层提供了点对点的通信服务&#xff0c;努力交付数据报&#xff0c;但…

鸿蒙OpenHarmony开发实战-0开始做游戏渲染引擎

首先实现了一个通用的画廊组件来作为练手项目&#xff0c;它主要使用了四个基础组件和容器组件&#xff1a; 我们放置一个按钮来触发 showGallery 方法&#xff0c;该方法控制 panel 弹出式组件的显示和隐藏&#xff0c;这里的 div 和 button 标签就是 hml 内置的组件&#xf…

docker安装postgresql15或者PG15

1. 查询版本 docker search postgresql docker pull postgres:15.3 # 也可以拉取其他版本2.运行容器并挂载数据卷 mkdir -p /data/postgresql docker run --name postgres \--restartalways \-e POSTGRES_PASSWORDpostgresql \-p 5433:5432 \-v /data/postgresql:/var/lib/p…

防火墙未开端口导致zookeeper集群异常,kafka起不来

转载说明&#xff1a;如果您喜欢这篇文章并打算转载它&#xff0c;请私信作者取得授权。感谢您喜爱本文&#xff0c;请文明转载&#xff0c;谢谢。 问题描述&#xff1a; 主机信息&#xff1a; IPhostname10.0.0.10host1010.0.0.12host1210.0.0.13host13 在这三台主机上部署…

clickhouseSQL日期相关

1. 毫秒级时间戳转日期/小时 --13位时间戳转具体时间 toDateTime(report_time / 1000) as _c00 -- 获取时间戳对应的时间点整点(结果&#xff1a;%Y-%m-%d %H:00:00.0) eg&#xff1a;2022-09-28 23:00:00.0 toStartOfHour(toDateTime(report_time / 1000)) AS _10-- 获取时间…

认识SpringBoot项目中的Starter

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a; 循序渐进学SpringBoot ✨特色专栏&…

基于 EigenFaces 的人脸检测

EigenFaces概述 EigenFaces 人脸检测是一种从主成分分析&#xff08;Principal Component Analysis&#xff0c;PCA&#xff09;中导出的人脸识别和描述技术。 特征脸方法就是从大量的人脸图像中&#xff0c;寻找出人脸的共性。将眼睛、面颊、下颌样板采集协方差矩阵的特征向…