使用android SpannableStringBuilder实现图文混排,查看更多

本文介绍如何在 Android 中使用 SpannableStringBuilder 和 ImageSpan 实现文本与图片的混合展示,包括文字高亮、添加下划线等效果,并提供代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

项目开发中需要实现这种效果

多余两行,两行最后是省略号,省略号后面是下拉更多

之前用过的是Html.fromHtml去处理图文混排的,仅仅是文字后图片或者文字颜色字体什么的,

但是这里需要在最后文字的省略号后面添加图片。

直接上代码吧,代码注释很多,慢慢研究

private void toggleEllipsize(final TextView tv,final String desc){
    if(desc == null){
    return;
    }
    tv.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
      
      @Override
      public void onGlobalLayout() {
        boolean isEllipsized = (tv.getTag()==null||tv.getTag().equals(false))?false:(Boolean)tv.getTag();
        if(isEllipsized){
          tv.setTag(false);
        tv.setText(desc);
        }else{
        tv.setTag(true);
        int paddingLeft = tv.getPaddingLeft();
        int paddingRight = tv.getPaddingRight();
        TextPaint paint = tv.getPaint();
        float moreText = tv.getTextSize()*3;
        float availableTextWidth = (tv.getWidth()-paddingLeft-paddingRight)*2-moreText;
        
        CharSequence ellipsizeStr = TextUtils.ellipsize(desc,paint,availableTextWidth,TextUtils.TruncateAt.END);
        if(ellipsizeStr.length()<desc.length()){
          /*String html = "<img src='game_info_lookmore'/>";
          CharSequence charSequence = Html.fromHtml(html, new ImageGetter() {

            @Override
            public Drawable getDrawable(String source) {
              Drawable drawable = getResources().getDrawable(
                  getResourceId(source));
              drawable.setBounds(
                  0,
                  0,
                  drawable.getIntrinsicWidth()
                      - DensityUtil.dip2px(GridGameInfoActivity.this, 3),
                  drawable.getIntrinsicHeight()
                      - DensityUtil.dip2px(GridGameInfoActivity.this, 1));
              return drawable;
            }
          }, null);
          ellipsizeStr = ellipsizeStr.toString() + charSequence.toString();*/
          
          
          CharSequence temp = ellipsizeStr+".";
          SpannableStringBuilder ssb = new SpannableStringBuilder(temp);
          Drawable dd = getResources().getDrawable(R.drawable.game_info_lookmore);
          dd.setBounds(0, 0, dd.getIntrinsicWidth(), dd.getIntrinsicHeight());
          ImageSpan is = new ImageSpan(dd, ImageSpan.ALIGN_BASELINE);
          ssb.setSpan(is, temp.length()-1, temp.length(), Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
          
//					int yellow = getResources().getColor(R.color.red);
//					ssb.setSpan(new ForegroundColorSpan(yellow),ssb.length()-2,ssb.length(),Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
          tv.setText(ssb);
          tv.setMovementMethod(LinkMovementMethod.getInstance());
        }else{
          tv.setText(desc);
        }
        }
        if(Build.VERSION.SDK_INT>=16){
          tv.getViewTreeObserver().removeOnGlobalLayoutListener(this);
        }else{
          tv.getViewTreeObserver().removeGlobalOnLayoutListener(this);
        }
      }
    });
    }
主要是通过SpannableStringBuilder把省略的文字和最后的图片给拼接起来。也可以最后拼接的是文字,

不让... 更多

转篇博客:

Android spannableStringBuilder用法整理

spannableStringBuilder 用法详解:
 SpannableString ss = new SpannableString("红色打电话斜体删除线绿色下划线图片:.");  
   //用颜色标记文本
   ss.setSpan(new ForegroundColorSpan(Color.RED), 0, 2,  
     //setSpan时需要指定的 flag,Spanned.SPAN_EXCLUSIVE_EXCLUSIVE(前后都不包括).
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //用超链接标记文本
   ss.setSpan(new URLSpan("tel:4155551212"), 2, 5,  
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //用样式标记文本(斜体)
   ss.setSpan(new StyleSpan(Typeface.BOLD_ITALIC), 5, 7,  
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //用删除线标记文本
   ss.setSpan(new StrikethroughSpan(), 7, 10,  
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //用下划线标记文本
   ss.setSpan(new UnderlineSpan(), 10, 16,  
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //用颜色标记
   ss.setSpan(new ForegroundColorSpan(Color.GREEN), 10, 13,  
     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
   //获取Drawable资源
   Drawable d = getResources().getDrawable(R.drawable.icon);  
   d.setBounds(0, 0, d.getIntrinsicWidth(), d.getIntrinsicHeight());
   //创建ImageSpan
   ImageSpan span = new ImageSpan(d, ImageSpan.ALIGN_BASELINE);
   //用ImageSpan替换文本
   ss.setSpan(span, 18, 19, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);  
   txtInfo.setText(ss);
   txtInfo.setMovementMethod(LinkMovementMethod.getInstance()); //实现文本的滚动  
通常用于显示文字,但有时候也需要在文字中夹杂一些图片,比如QQ中就可以使用表情图片,又比如需要的文字高亮显示等等,如何在android中也做到这样呢? 
记得android中有个android.text包,这里提供了对文本的强大的处理功能。 
添加图片主要用SpannableString和ImageSpan类:
 
     Drawable drawable = getResources().getDrawable(id);  
  drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());  
  //需要处理的文本,[smile]是需要被替代的文本  
  SpannableString spannable = new SpannableString(getText().toString()+"[smile]");  
  //要让图片替代指定的文字就要用ImageSpan  
  ImageSpan span = new ImageSpan(drawable, ImageSpan.ALIGN_BASELINE);  
  //开始替换,注意第2和第3个参数表示从哪里开始替换到哪里替换结束(start和end)  
       //最后一个参数类似数学中的集合,[5,12)表示从5到12,包括5但不包括12  
  spannable.setSpan(span, getText().length(),getText().length()+"[smile]".length(), Spannable.SPAN_INCLUSIVE_EXCLUSIVE);    
  setText(spannable);  
 
将需要的文字高亮显示: 
 
 
 
public void highlight(int start,int end){  
  SpannableStringBuilder spannable=new SpannableStringBuilder(getText().toString());//用于可变字符串  
  ForegroundColorSpan span=new ForegroundColorSpan(Color.RED);  
  spannable.setSpan(span, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  setText(spannable);  
    }  
   
加下划线: 
 
 
 
public void underline(int start,int end){  
  SpannableStringBuilder spannable=new SpannableStringBuilder(getText().toString());  
  CharacterStyle span=new UnderlineSpan();  
  spannable.setSpan(span, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  setText(spannable);  
    }  
   
组合运用:
 
 
 
SpannableStringBuilder spannable=new SpannableStringBuilder(getText().toString());  
  CharacterStyle span_1=new StyleSpan(android.graphics.Typeface.ITALIC);  
  CharacterStyle span_2=new ForegroundColorSpan(Color.RED);  
  spannable.setSpan(span_1, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  spannable.setSpan(span_2, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  setText(spannable);  
   
案例:带有\n换行符的字符串都可以用此方法显示2种颜色
 
 
 
    /** 
     * 带有\n换行符的字符串都可以用此方法显示2种颜色 
     * @param text 
     * @param color1 
     * @param color2 
     * @return 
     */  
    public SpannableStringBuilder highlight(String text,int color1,int color2,int fontSize){  
  SpannableStringBuilder spannable=new SpannableStringBuilder(text);//用于可变字符串  
  CharacterStyle span_0=null,span_1=null,span_2;  
  int end=text.indexOf("\n");  
  if(end==-1){//如果没有换行符就使用第一种颜色显示  
      span_0=new ForegroundColorSpan(color1);  
      spannable.setSpan(span_0, 0, text.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  }else{  
      span_0=new ForegroundColorSpan(color1);  
      span_1=new ForegroundColorSpan(color2);  
      spannable.setSpan(span_0, 0, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
      spannable.setSpan(span_1, end+1, text.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
        
      span_2=new AbsoluteSizeSpan(fontSize);//字体大小  
      spannable.setSpan(span_2, end+1, text.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);  
  }  
  return spannable;  
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值