经典使用场景
SpannableStringBuilder needStartSSB = new SpannableStringBuilder("需要");
SpannableString mSpannableString = new SpannableString("多少");

//颜色
ForegroundColorSpan colorSpan = new ForegroundColorSpan(0xffffa726);
mSpannableString.setSpan(colorSpan, 0, mSpannableString.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
//大小
AbsoluteSizeSpan absoluteSizeSpan = new AbsoluteSizeSpan(ApplicationUtil.dip2px(15));
mSpannableString.setSpan(absoluteSizeSpan, 0, mSpannableString.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);

needStartSSB.append(mSpannableString).append("财富值");  

富文本类简介

设置富文本信息的几种方式:

  • 1、拼接多个TextView,如果此富文本信息中多个字段是独立的,可以采取这种方式,否则这种方式是繁琐且不合理的
  • 2、使用Html.fromHtml(),此方法可以转换html格式的字符串为Android支持的字符串,简单的颜色,下划线,粗体及链接是比较方便的,但是有些标签可能会不兼容,另外注意某些特殊符号,特别是双引号,需要在前面添加转义字符。
  • 3、可以在布局中直接定义Html类型的富文本样式,注意,只能在values/strings文件中定义,否则不起作用,另外,这种方式支持的标签数比上面代码中的还少,但是对于一些简单的样式,特别是下划线,这种方式是最方便的。
  • 4、使用SpannableString和SpannableStringBuilder类,使用这种方式可以灵活的设置复杂的富文本样式,但是代码量偏多,下面详细介绍这种方式。
简介
  • SpannableString和SpannableStringBuilder相比,SpannableString像String一样,构造对象的时候传入一个String之后无法更改String的内容,也无法拼接多个 SpannableString;而SpannableStringBuilder则像是StringBuilder,它可以通过其append()方法来拼接多个SpannableString。
  • 因为Spannable实现了CharSequence接口,所以可以直接把SpannableString和SpannableStringBuilder通过TextView.setText(CharSequence)设置给TextView
  • 注意不能将SpannableString或SpannableStringBuilder通过toString或【+"字符串"】,否则富文本就变成纯文本了。
  • 另外,由于TextView具有tv.append(CharSequence)方法,所以在给TextView设置多个富文本信息时,也可采用append()多个SpannableString的方式,具体详见示例代码。
API的Class General Hierarchy:
     

设置样式setSpan

SpannableString和SpannableStringBuilder都有一个设置Span的方法:setSpan(Object what, int start, int end, int flags);

  • 参数what是要设置的Span ;
  • start和end则是标识此String中需要应用Span的起始位置(0表示第一个字符,包含开始不包含结尾),注意:范围超出文字大小时会报异常
  • 而 flags是用来标识在 Span 范围内的文本【前后】输入【新】的字符时是否把它们也应用这个效果,我们一般都是用 Spanned.SPAN_EXCLUSIVE_EXCLUSIVE
自带的 flags有(其中EXCLUSIVE代表不包含,INCLUSIVE代表包含)
  • Spanned.SPAN_EXCLUSIVE_EXCLUSIVE --- 不包含两端start和end所在的端点
  • Spanned.SPAN_EXCLUSIVE_INCLUSIVE --- 不包含端start,但包含end所在的端点
  • Spanned.SPAN_INCLUSIVE_EXCLUSIVE --- 包含端start,但不包含end所在的端点
  • Spanned.SPAN_INCLUSIVE_INCLUSIVE--- 包含两端start和end所在的端点
 常见的Span类型
  • ForegroundColorSpan 文本颜色
  • URLSpan 文本超链接,要设置setMovementMethod(LinkMovementMethod.getInstance())才可响应点击
  • BackgroundColorSpan 背景色
  • UnderlineSpan 下划线
  • AbsoluteSizeSpan 绝对大小(文本字体),一般是将dp2px转换后的值传给他
  • ClickableSpan 文本可点击,有点击事件
  • MaskFilterSpan 修饰效果,如模糊(BlurMaskFilter)、浮雕(EmbossMaskFilter)
  • MetricAffectingSpan 父类,一般不用
  • RasterizerSpan 光栅效果
  • StrikethroughSpan 删除线(中划线)
  • SuggestionSpan 相当于占位符
  • DynamicDrawableSpan 设置图片,基于文本基线或底部对齐。
  • ImageSpan 图片
  • RelativeSizeSpan 文本字体相对大小,参数是相对于默认字体大小的倍数
  • ReplacementSpan 父类,一般不用
  • ScaleXSpan 基于x轴缩放
  • StyleSpan 字体样式:粗体、斜体等,参数是android.graphics.Typeface里面定义的常量
  • SubscriptSpan 下标(数学公式会用到)
  • SuperscriptSpan 上标(数学公式会用到)
  • TextAppearanceSpan 文本外貌(包括字体、大小、样式和颜色)
  • TypefaceSpan 文本字体,参数是字体的名字比如“sans", "sans-serif"等

演示代码

public class MainActivity extends Activity {

    private TextView tv1;
    private TextView tv2;
    private TextView tv3;
    private TextView tv4;
    private TextView tv5;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tv1 = (TextView) findViewById(R.id.tv1);
        tv2 = (TextView) findViewById(R.id.tv2);
        tv3 = (TextView) findViewById(R.id.tv3);
        tv4 = (TextView) findViewById(R.id.tv4);
        tv5 = (TextView) findViewById(R.id.tv5);
        setForegroundColor();
        setBackgroundColor();
        setStyle();
        setSize();
        setLink();
    }
    //文字颜色,标准写法一
    private void setForegroundColor() {
        SpannableString spanString = new SpannableString("文字颜色 ");
        ForegroundColorSpan span = new ForegroundColorSpan(Color.BLUE);//常用values/colors中定义的颜色
        spanString.setSpan(span, 0, spanString.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        //tv1.setText(spanString, BufferType.SPANNABLE);//一般不用设置后面的参数,但有时必须使用这种格式
        tv1.setText(spanString);
    }
    //背景色
    private void setBackgroundColor() {
        tv2.setText("应付:");
        SpannableString spanString = new SpannableString("888");
        BackgroundColorSpan span = new BackgroundColorSpan(0x88008800);
        spanString.setSpan(span, 0, spanString.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        //    tv2.append(spanString + " 元");//这样是不行的,因为【spanString + " 元"】的结果是一个没有格式的字符串
        tv2.append(spanString);//这样是可以的
        tv2.append(" 元");
    }
    //粗体、斜体、删除线、下划线,使用多个SpannableString设置多个富文本样式
    private void setStyle() {
        tv3.setText("样式:");
        SpannableString mSpannableString1 = new SpannableString("粗体 ");
        StyleSpan mStyleSpan = new StyleSpan(Typeface.BOLD);
        mSpannableString1.setSpan(mStyleSpan, 0, mSpannableString1.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        tv3.append(mSpannableString1);
        SpannableString mSpannableString2 = new SpannableString("删除线 ");
        StrikethroughSpan mStrikethroughSpan = new StrikethroughSpan();
        mSpannableString2.setSpan(mStrikethroughSpan, 0, mSpannableString2.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        tv3.append(mSpannableString2);
        SpannableString mSpannableString3 = new SpannableString("下划线");
        UnderlineSpan mUnderlineSpan = new UnderlineSpan();
        mSpannableString3.setSpan(mUnderlineSpan, 0, mSpannableString3.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        tv3.append(mSpannableString3);
    }
    //文字链接,使用一个SpannableStringBuilder及多个SpannableString设置多个富文本样式
    private void setLink() {
        SpannableStringBuilder mSpannableStringBuilder = new SpannableStringBuilder("链接:电话 ");
        URLSpan mUrlSpan = new URLSpan("tel:110");//会把参数通过intent的putExtra的参数传进去,注意格式!
        mSpannableStringBuilder.setSpan(mUrlSpan, 3, mSpannableStringBuilder.length() - 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        SpannableString mSpannableString2 = new SpannableString("网址 ");
        URLSpan mUrlSpan2 = new URLSpan("http://www.baidu.com");
        mSpannableString2.setSpan(mUrlSpan2, 0, mSpannableString2.length() - 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        mSpannableStringBuilder.append(mSpannableString2);
        SpannableString mSpannableString3 = new SpannableString("邮箱 ");
        URLSpan mUrlSpan3 = new URLSpan("mailto:bqt@sina.com");
        mSpannableString3.setSpan(mUrlSpan3, 0, mSpannableString3.length() - 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        mSpannableStringBuilder.append(mSpannableString3);
        SpannableString mSpannableString4 = new SpannableString("短信");
        URLSpan mUrlSpan4 = new URLSpan("smsto:10086");
        mSpannableString4.setSpan(mUrlSpan4, 0, mSpannableString4.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        mSpannableStringBuilder.append(mSpannableString4);
        tv4.setText(mSpannableStringBuilder);
        tv4.setMovementMethod(LinkMovementMethod.getInstance());//要设置这个东东才能点击
    }
    //字号
    private void setSize() {
        tv5.setText("字号:");
        SpannableStringBuilder mSpannableStringBuilder = new SpannableStringBuilder("相对大小 ");
        RelativeSizeSpan mRelativeSizeSpan = new RelativeSizeSpan(0.5f);
        mSpannableStringBuilder.setSpan(mRelativeSizeSpan, 0, mSpannableStringBuilder.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        SpannableString mSpannableString = new SpannableString("绝对大小");
        AbsoluteSizeSpan span = new AbsoluteSizeSpan(dp2px(20));
        mSpannableString.setSpan(span, 0, mSpannableString.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        mSpannableStringBuilder.append(mSpannableString);
        tv5.append(mSpannableStringBuilder);
    }
    /** 
    * 根据手机的分辨率从 dp 的单位 转成为 px(像素) 
    */
    public int dp2px(float dpValue) {
        float scale = getResources().getDisplayMetrics().density;
        return (int) (dpValue * scale + 0.5f);
    }

}


演示布局


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#fff"
    android:orientation="vertical"
    android:padding="10dp" >
    <TextView
        android:id="@+id/tv1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
    <TextView
        android:id="@+id/tv2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
    <TextView
        android:id="@+id/tv3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
    <TextView
        android:id="@+id/tv4"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
    <TextView
        android:id="@+id/tv5"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

</LinearLayout>


图片--可以实现图文混排的效果

需求:在文字后面加上一个图标

tv = (TextView) findViewById(R.id.tv);

        String text = "哈哈★我能把★换成图片";
        SpannableStringBuilder builder = new SpannableStringBuilder(text);
        Pattern pattern = Pattern.compile("★");
        Matcher matcher = pattern.matcher(text);
        while (matcher.find()) {
            //不能在while循环外定义ImageSpan,否则只会把最后一个★替换掉
            builder.setSpan(new ImageSpan(this, R.drawable.ic_launcher), matcher.start(), matcher.end(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        }
        tv.setGravity(Gravity.CENTER);//无效!不知道怎么解决

tv.setText(builder);  


tv = (TextView) findViewById(R.id.tv);

        String text = "哈哈★我能把★换成图片";
        SpannableStringBuilder builder = new SpannableStringBuilder(text);
        Pattern pattern = Pattern.compile("★");
        Matcher matcher = pattern.matcher(text);
        Drawable drawable = getResources().getDrawable(R.drawable.ic_launcher);
        //必须为drawable设置边界(下面的就是标准格式),否则图片不会显示
        drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
        while (matcher.find()) {
            //不能在while循环外定义ImageSpan,否则只会把最后一个★替换掉
            builder.setSpan(new ImageSpan(drawable, ImageSpan.ALIGN_BASELINE), matcher.start(), matcher.end(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        }
        tv.setGravity(Gravity.CENTER);//无效!不知道怎么解决

tv.setText(builder);


富文本 SpannableString Span的更多相关文章

  1. Vue富文本编辑器(图片拖拽缩放)

    富文本编辑器(图片拖拽缩放) 需求: 根据业务要求,需要能够上传图片,且上传的图片能在移动端中占满屏幕宽度,故需要能等比缩放上传的图片,还需要能拖拽.缩放.改变图片大小.尝试多个第三方富文本编辑器,很 ...

  2. springmvc 后台实体类接受前端json字符串时,其中一个属性content 接受富文本内容时 标签<p>、<span> 这些标签丢失问题解决

    问题描述: 前端一个字段 <script id="editor" type="text/plain" name="content" s ...

  3. SpannableString富文本

    忍不住想吐槽这个类,这个类是要给文本设置不同的颜色.字体样式 例子:一句话中只有某几个文字想要设置成不同的颜色 起初写了一个函数setColorStyle(), public SpannableStr ...

  4. TextView展示富文本时emoj或图片和文字不对齐的解决方案

    在项目中,回复框.聊天界面的显示往往会有emoj或者图片,但是一个比较头疼的问题是,会出现emoj表情或者图片和文字的位置不对齐,总是有偏移,这样很影响用户体验的.下面会总结一下如何解决这个问题. 本 ...

  5. Android - 富文本编辑器

    Android富文本编辑器(一):基础知识 目前主流的基于Android富文本开发方式思路如下: 基于TextView图文混排 使用方式: TextView textView = new TextVi ...

  6. 个人网站对xss跨站脚本攻击(重点是富文本编辑器情况)和sql注入攻击的防范

    昨天本博客受到了xss跨站脚本注入攻击,3分钟攻陷--其实攻击者进攻的手法很简单,没啥技术含量.只能感叹自己之前竟然完全没防范. 这是数据库里留下的一些记录.最后那人弄了一个无限循环弹出框的脚本,估计 ...

  7. 富文本编辑器kindeditor配置

    <!--富文本编辑器kindeditor配置↓ --> <link type="text/css" rel="stylesheet" href ...

  8. 富文本编辑器防止xss注入javascript版

    富文本编辑器:ueditor 其实富文本编辑器已经有防止xss注入功能,但是你服务端程序在接收的时候在做一次转义,否则有可能然后前端验证直接提交数据导致被xss攻击. 为了节省后端程序开销则在前端 显 ...

  9. 对于MVC中应用百度富文本编辑器问题的解决办法

    1.对于应用富文本编辑器post提交表单内容提示有危险的解决办法: [ValidateInput(false)] //文本编辑器的表单提交不用提示危险 [HttpPost] public Action ...

随机推荐

  1. [转自已]Windos多个文件快速重命名说明+图解

    转自己以前的文章,给新博客带点气氛. 1.(复制的)比如在文件夹中包含yin.jpg.ye.jpg.zou.jpg三个文件,你希望将它们命名为"photo+数字"的文件名形式,那么 ...

  2. 在IE6/7/8下识别html5标签

    识别html5标签: html5添加了许多语义化的标签,比如<nav></nav>,<aside></aside>,<article>< ...

  3. python之路基础篇

    基础篇 1.Python基础之初识python 2.Python数据类型之字符串 3.Python数据类型之列表 4.Python数据类型之元祖 5.Python数据类型之字典 6.Python Se ...

  4. python中的几种遍历列表的方法比较

    python的内容非常丰富,给我们带来的便利很多,很多事情的表达方法有很大的多样性,比如我经常需要遍历一个列表,取它的下标和值,这个时候就有很多方法需要取舍一下才行. for循环遍历 l = [1,2 ...

  5. UVA - 12627 Erratic Expansion 奇怪的气球膨胀 (分治)

    紫书例题p245 Piotr found a magical box in heaven. Its magic power is that if you place any red balloon i ...

  6. UVA - 524 Prime Ring Problem(dfs回溯法)

    UVA - 524 Prime Ring Problem Time Limit:3000MS     Memory Limit:0KB     64bit IO Format:%lld & % ...

  7. Java中堆内存(heap)和栈内存(stack)的区别

    在Java代码中,常常会使用到这样的类的声明实例化: Person per = new Person(); //这其实是包含了两个步骤,声明和实例化 Person per = null; //声明一个 ...

  8. asp.net 后台使用js弹窗失效问题

    1.这些事件输出来前后都变成JS代码了,看到到这样的代码的了.会变成<script>alert('合同号XXX已存在')</script>首先后台调试一下看看Page.Clie ...

  9. 在vs中连接sql的几种连接方式

     sql身份验证:Data Source=.;Initial Catalog=DBName;UID=sa;Pwd=pwd windows身份验证:Data Source=.;Initial Catal ...

  10. Windows系统编程之进程间通信

    Windows系统编程之进程间通信作者:北极星2003来源:看雪论坛(www.pediy.com)Windows 的IPC(进程间通信)机制主要是异步管道和命名管道.(至于其他的IPC方式,例如内存映 ...