文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

怎么在Android中利用SpannableString对内容进行格式化

2023-05-31 10:17

关注

这篇文章将为大家详细讲解有关怎么在Android中利用SpannableString对内容进行格式化,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

要实现的效果:

  1. 将话题进行变色并且可以点击提示对应的话题文本内容

  2. 将图片表情替换掉对应的表情关键字显示

  3. 将链接地址替换成一个链接的图片和”网页链接”四个字显示

  4. 将@的用户进行变色并且可以点击提示对应的话题文本内容

需要:

  1. 使用正则表达式提取文本内对应的”话题”、”表情”、”网页链接”、以及”@用户”内容

  2. 使用 SpannableString 格式化提取到的文本

  3. 给格式化的部分添加点击事件

定义正则表达式

首先定义”话题”、”表情”、”网页链接”、以及”@用户”对应的正则表达式和对应的 Pattern。SCHEME 下文会提到具体的用处的。

public class WeiboPattern { // #话题# public static final String REGEX_TOPIC = "#[\\p{Print}\\p{InCJKUnifiedIdeographs}&&[^#]]+#"; // [表情] public static final String REGEX_EMOTION = "\\[(\\S+?)\\]"; // url public static final String REGEX_URL = "http://[a-zA-Z0-9+&@#/%?=~_\\\\-|!:,\\\\.;]*[a-zA-Z0-9+&@#/%=~_|]"; // @人 public static final String REGEX_AT = "@[\\w\\p{InCJKUnifiedIdeographs}-]{1,26}";  public static final Pattern PATTERN_TOPIC = Pattern.compile(REGEX_TOPIC); public static final Pattern PATTERN_EMOTION = Pattern.compile(REGEX_EMOTION); public static final Pattern PATTERN_URL = Pattern.compile(REGEX_URL); public static final Pattern PATTERN_AT = Pattern.compile(REGEX_AT); public static final String SCHEME_TOPIC = "topic:"; public static final String SCHEME_URL = "url:"; public static final String SCHEME_AT = "at:";}

提取匹配部分并使用 SpannableString 格式化

我将此过程写到一个方法内了,下面直接上代码,代码中有详细的注释解释:

public static SpannableStringBuilder formatWeiBoContent(Context context, String source, TextView textView) { // 获取到 TextView 的文字大小,后面的 ImageSpan 需要用到该值 int textSize = (int) textView.getTextSize(); // 若要部分 SpannableString 可点击,需要如下设置 textView.setMovementMethod(LinkMovementMethod.getInstance()); // 将要格式化的 String 构建成一个 SpannableStringBuilder SpannableStringBuilder value = new SpannableStringBuilder(source); // 使用正则匹配话题 Linkify.addLinks(value, WeiboPattern.PATTERN_TOPIC, WeiboPattern.SCHEME_TOPIC); // 使用正则匹配链接 Linkify.addLinks(value, WeiboPattern.PATTERN_URL, WeiboPattern.SCHEME_URL); // 使用正则匹配@用户 Linkify.addLinks(value, WeiboPattern.PATTERN_AT, WeiboPattern.SCHEME_AT); // 自定义的匹配部分的点击效果 MyClickableSpan clickSpan; // 获取上面到所有 addLinks 后的匹配部分(这里一个匹配项被封装成了一个 URLSpan 对象) URLSpan[] urlSpans = value.getSpans(0, value.length(), URLSpan.class); // 遍历所有的 URLSpan for (final URLSpan urlSpan : urlSpans) { // 点击匹配部分效果  clickSpan = new MyClickableSpan() {   @Override   public void onClick(View view) {    ToastUtils.makeShort(urlSpan.getURL());   }  };  // 话题  if (urlSpan.getURL().startsWith(WeiboPattern.SCHEME_TOPIC)) {   int start = value.getSpanStart(urlSpan);   int end = value.getSpanEnd(urlSpan);   value.removeSpan(urlSpan);   // 格式化话题部分文本   value.setSpan(clickSpan, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);  }  // @用户  if (urlSpan.getURL().startsWith(WeiboPattern.SCHEME_AT)) {   int start = value.getSpanStart(urlSpan);   int end = value.getSpanEnd(urlSpan);   value.removeSpan(urlSpan);   // 格式化@用户部分文本   value.setSpan(clickSpan, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);  }  // 链接  if (urlSpan.getURL().startsWith(WeiboPattern.SCHEME_URL)) {   int start = value.getSpanStart(urlSpan);   int end = value.getSpanEnd(urlSpan);   value.removeSpan(urlSpan);   SpannableStringBuilder urlSpannableString = getUrlTextSpannableString(context, urlSpan.getURL(), textSize);   value.replace(start, end, urlSpannableString);   // 格式化链接部分文本   value.setSpan(clickSpan, start, start + urlSpannableString.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);  } } // 表情需要单独格式化 Matcher emotionMatcher = WeiboPattern.PATTERN_EMOTION.matcher(value); while (emotionMatcher.find()) {  String emotion = emotionMatcher.group();  int start = emotionMatcher.start();  int end = emotionMatcher.end();  int resId = EmotionUtils.getImageByName(emotion);  if (resId != -1) { // 表情匹配   L.e("find emotion: " + emotion);   Drawable drawable = context.getResources().getDrawable(resId);   drawable.setBounds(0, 0, (int) (textSize * 1.3), (int) (textSize * 1.3));   // 自定义的 VerticalImageSpan ,可解决默认的 ImageSpan 不垂直居中的问题   VerticalImageSpan imageSpan = new VerticalImageSpan(drawable);   value.setSpan(imageSpan, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  } } return value;}
private static SpannableStringBuilder getUrlTextSpannableString(Context context, String source, int size) { SpannableStringBuilder builder = new SpannableStringBuilder(source); String prefix = " "; builder.replace(0, prefix.length(), prefix); Drawable drawable = context.getResources().getDrawable(R.drawable.ic_status_link); drawable.setBounds(0, 0, size, size); builder.setSpan(new VerticalImageSpan(drawable), prefix.length(), source.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); builder.append(" 网页链接"); return builder;}

getUrlTextSpannableString() :方法是用来返回一个图标+”网页链接” SpannableString,用于替换链接文本

上面将”话题”、”表情”、”网页链接”都用了addLinks方法来标记的,然后统一处理。表情则是单独处理的。

表情则使用如下方法事先做好映射:

public class EmotionUtils { public static LinkedHashMap<String, Integer> sMap; static {  sMap = new LinkedHashMap<>();  sMap.put("[doge]", R.drawable.d_doge);  sMap.put("[污]", R.drawable.d_wu); } public static int getImageByName(String name) {  Integer integer = sMap.get(name);  return integer == null ? -1 : integer; }}

还有刚才说到的自定义 MyClickableSpan 修改默认的样式:

public class MyClickableSpan extends ClickableSpan { @Override public void onClick(View view) { } @Override public void updateDrawState(TextPaint ds) {  super.updateDrawState(ds);  ds.setColor(0xff03A9F4);  ds.setUnderlineText(false); }}

另外,由于默认的 ImageSpan 在 TextView 有使用android:lineSpacingExtra属性时,不会垂直居中,所以使用到了网上的一个继承自 ImageSpan 的 VerticalImageSpan 可以做到保持图片在 TextView 内保持垂直居中:

public class VerticalImageSpan extends ImageSpan { public VerticalImageSpan(Drawable drawable) {  super(drawable); }  @Override public int getSize(Paint paint, CharSequence text, int start, int end,      Paint.FontMetricsInt fontMetricsInt) {  Drawable drawable = getDrawable();  Rect rect = drawable.getBounds();  if (fontMetricsInt != null) {   Paint.FontMetricsInt fmPaint = paint.getFontMetricsInt();   int fontHeight = fmPaint.descent - fmPaint.ascent;   int drHeight = rect.bottom - rect.top;   int centerY = fmPaint.ascent + fontHeight / 2;   fontMetricsInt.ascent = centerY - drHeight / 2;   fontMetricsInt.top = fontMetricsInt.ascent;   fontMetricsInt.bottom = centerY + drHeight / 2;   fontMetricsInt.descent = fontMetricsInt.bottom;  }  return rect.right; }  @Override public void draw(Canvas canvas, CharSequence text, int start, int end,      float x, int top, int y, int bottom, Paint paint) {  Drawable drawable = getDrawable();  canvas.save();  Paint.FontMetricsInt fmPaint = paint.getFontMetricsInt();  int fontHeight = fmPaint.descent - fmPaint.ascent;  int centerY = y + fmPaint.descent - fontHeight / 2;  int transY = centerY - (drawable.getBounds().bottom - drawable.getBounds().top) / 2;  canvas.translate(x, transY);  drawable.draw(canvas);  canvas.restore(); }}

然后直接调用该方法格式化:

mTextView.setText(formatWeiBoContent(this,mTextView.getText().toString(),mTextView))

关于怎么在Android中利用SpannableString对内容进行格式化就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯