给TextView插上SpannableString的翅膀

SpannableString(标签文本),可以用来显示复合文本,我们可以通过SpannableString给文本设置各种各样的样式。比如部分文本变色,商城打折的删除线,粗体,斜体,不同大小的文字,文字混合表情,文字点击链接等。TextView通过设置SpannableString后功能就异常强大了,在自定义的Span的情况下,可以说就要上天了。

完整代码请戳github代码直通车

认识SpannableString:

在TextView的public final void setText(CharSequence text)方法中,设置文本,那么这个我们每天都要用的CharSequence到底是什么呢?看Google的官方文档:

给TextView插上SpannableString的翅膀_第1张图片
image.png

SpannableString继承于CharSequence,所以通过textView.setText(spannableString)的方式使用。SpannableString构造方法参数传入需要设置样式的文本。

public void setSpan(Object what, int start, int end, int flags)

what为各种类型Span,start为起始位置,end为结束位置,flags为开闭区间的标致。

flags类型:
  • Spanned.SPAN_EXCLUSIVE_EXCLUSIVE 开开区间,如(0,1)
  • Spanned.SPAN_INCLUSIVE_EXCLUSIVE 闭开区间,如[0,1)
  • Spanned.SPAN_EXCLUSIVE_INCLUSIVE 开闭区间,如(0,1]
  • Spanned.SPAN_INCLUSIVE_INCLUSIVE 闭闭区间,如[0,1]
使用步骤:
  1. 创建SpannableString,构造方法设置文本
  2. 创建Span,SpannableString.setSpan()
  3. textView.setText(spannableString)
  • ForegroundColorSpan 设置文字前景色(文字颜色)
给TextView插上SpannableString的翅膀_第2张图片
image.png
    public static SpannableString getForegroundColorSpan(Context context){
        SpannableString spannableString = new SpannableString("我已同意抖音使用协议");
        ForegroundColorSpan foregroundColorSpan = new ForegroundColorSpan(context.getResources().getColor(R.color.colorPrimary));
        spannableString.setSpan(foregroundColorSpan,4,spannableString.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }
  • BackgroundColorSpan 设置文字背景色


    给TextView插上SpannableString的翅膀_第3张图片
    image.png
    public static SpannableString getBackgroundColorSpan(Context context){
        SpannableString spannableString = new SpannableString("你看我头像牛逼不?");
        BackgroundColorSpan backgroundColorSpan = new BackgroundColorSpan(context.getResources().getColor(R.color.colorAccent));
        spannableString.setSpan(backgroundColorSpan,3,5,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }
  • StrikethroughSpan 设置删除线


    给TextView插上SpannableString的翅膀_第4张图片
    image.png
    public static SpannableString getStrikethroughSpan(){
        SpannableString spannableString = new SpannableString("尤龙的传人");
        StrikethroughSpan strikethroughSpan = new StrikethroughSpan();
        spannableString.setSpan(strikethroughSpan,0,1,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }
  • UnderlineSpan 设置下划线


    给TextView插上SpannableString的翅膀_第5张图片
    image.png
    public static SpannableString getUnderlineSpan(){
        SpannableString spannableString = new SpannableString("这里是下划线");
        UnderlineSpan underlineSpan = new UnderlineSpan();
        spannableString.setSpan(underlineSpan,3,6,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }
  • ScaleXSpan 设置X轴方向拉伸,ScaleXSpan构造方法参数传缩放倍数


    给TextView插上SpannableString的翅膀_第6张图片
    image.png
    public static SpannableString getScaleXSpan(){
        SpannableString spannableString = new SpannableString("媳妇你长胖了");
        ScaleXSpan scaleXSpan = new ScaleXSpan(2);
        spannableString.setSpan(scaleXSpan,0,spannableString.length(),Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }
  • RelativeSizeSpan 设置文字比例大小,原文字大小基数为1,构造函数参数为float值,缩放为原来比例,给文字依序设置不同大小。


    给TextView插上SpannableString的翅膀_第7张图片
    image.png
    public static SpannableString getRelativeSizeSpan(){
        SpannableString spannableString = new SpannableString("我心情忐忑不安七上八下");
        RelativeSizeSpan sizeSpan1 = new RelativeSizeSpan(1.2f);
        RelativeSizeSpan sizeSpan2 = new RelativeSizeSpan(1.4f);
        RelativeSizeSpan sizeSpan3 = new RelativeSizeSpan(1.6f);
        RelativeSizeSpan sizeSpan4 = new RelativeSizeSpan(1.8f);

        spannableString.setSpan(sizeSpan1,0,1,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,1,2,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,2,3,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan4,3,4,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,4,5,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,5,6,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan1,6,7,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,7,8,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan4,9,10,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);

        return spannableString;
    }
  • SuperscriptSpan 设置上标,给平方这个2设置为上标,并且文字缩小。当然这个在望京买套房是我奢侈的梦想。


    给TextView插上SpannableString的翅膀_第8张图片
    image.png
    public static SpannableString getSuperscriptSpan(){
        SpannableString spannableString = new SpannableString("刚在北京望京买了套1202m的房子");
        SuperscriptSpan superscriptSpan = new SuperscriptSpan();
        RelativeSizeSpan sizeSpan = new RelativeSizeSpan(0.7f);
        spannableString.setSpan(superscriptSpan,12,13,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan,12,13,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }
  • SubscriptSpan设置下标,给这个2设置为下标并缩小。


    给TextView插上SpannableString的翅膀_第9张图片
    image.png
    public static SpannableString getSubscriptSpan(){
        SpannableString spannableString = new SpannableString("水分子化学式为H20");
        SubscriptSpan subscriptSpan = new SubscriptSpan();
        RelativeSizeSpan sizeSpan = new RelativeSizeSpan(0.7f);
        spannableString.setSpan(subscriptSpan,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }
  • StyleSpan设置粗体、斜体样式:Typeface.BOLD为粗体字,Typeface.ITALIC为斜体字类型。


    给TextView插上SpannableString的翅膀_第10张图片
    image.png
    public static SpannableString getStyleSpan(){
        SpannableString spannableString = new SpannableString("身正不怕影子歪");
        StyleSpan styleSpanBold = new StyleSpan(Typeface.BOLD);
        StyleSpan styleSpanitalic = new StyleSpan(Typeface.ITALIC);
        spannableString.setSpan(styleSpanBold,0,4,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        spannableString.setSpan(styleSpanitalic,4,6,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }
  • ImageSpan 图文混合,比如在聊天中带有表情图标。这个是文字替换功能,将对应位置的字符替换为图标,这里将文本的空格替换为fx图标;drawable.setBounds()可设置图标的大小。


    给TextView插上SpannableString的翅膀_第11张图片
    image.png
    public static SpannableString getImageSpan(Context context){
        SpannableString spannableString = new SpannableString("芭芭拉小魔仙 魔法棒");
        Drawable drawable = context.getResources().getDrawable(R.mipmap.fx);
        drawable.setBounds(0,0,70,70);
        ImageSpan imageSpan = new ImageSpan(drawable);
        spannableString.setSpan(imageSpan,6,7,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }
  • ClickableSpan可点击标签,textview需设置:红色带下滑先的“哪里”,就是可点击范围,ClickableSpan自带点击回调。


    给TextView插上SpannableString的翅膀_第12张图片
    giphy (1).gif

tvClick.setMovementMethod(LinkMovementMethod.getInstance())方法,textview才会响应点击事件
//设置点击后文字没有背景
textView.setHighlightColor(getResources().getColor(android.R.color.transparent));

    public static SpannableString getClickableSpan(final Context context){
        SpannableString spannableString = new SpannableString("哪里不会点哪里,so easy!");
        ClickableSpan clickableSpan = new ClickableSpan() {
            @Override
            public void onClick(View widget) {
                Toast.makeText(context,"你戳中我了",Toast.LENGTH_SHORT).show();
            }

            @Override
            public void updateDrawState(TextPaint ds) {
                //设置去掉点击文字下划线
                ds.setUnderlineText(false);
            }
        };
        spannableString.setSpan(clickableSpan,5,7,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }
  • URLSpan跳转链接,可以设置为打电话,发短信,发邮件,打开网页等功能;类似于Intent的打电话,发短信,发邮件,打开网页的功能。


    给TextView插上SpannableString的翅膀_第13张图片
    giphy.gif
    public static SpannableString getURLSpan(){
        SpannableString spannableString = new SpannableString("打电话,发短信,发邮件,打开网页");
        spannableString.setSpan(new URLSpan("tel:10086"), 0, 3, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("smsto:10086"), 4, 7, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("mailto:[email protected]"), 8, 11, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("http://www.jianshu.com"), 12, 16, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

完整SpanUtils样式类:

import android.content.Context;
import android.graphics.Typeface;
import android.graphics.drawable.Drawable;
import android.text.SpannableString;
import android.text.Spanned;
import android.text.style.BackgroundColorSpan;
import android.text.style.ClickableSpan;
import android.text.style.ForegroundColorSpan;
import android.text.style.ImageSpan;
import android.text.style.RelativeSizeSpan;
import android.text.style.ScaleXSpan;
import android.text.style.StrikethroughSpan;
import android.text.style.StyleSpan;
import android.text.style.SubscriptSpan;
import android.text.style.SuperscriptSpan;
import android.text.style.URLSpan;
import android.text.style.UnderlineSpan;
import android.view.View;
import android.widget.Toast;

/**
 * Created by libo on 2017/12/12.
 */

public class SpanUtils {

    public static SpannableString getForegroundColorSpan(Context context){
        SpannableString spannableString = new SpannableString("我已同意抖音使用协议");
        ForegroundColorSpan foregroundColorSpan = new ForegroundColorSpan(context.getResources().getColor(R.color.colorPrimary));
        spannableString.setSpan(foregroundColorSpan,4,spannableString.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }

    public static SpannableString getBackgroundColorSpan(Context context){
        SpannableString spannableString = new SpannableString("你看我头像牛逼不?");
        BackgroundColorSpan backgroundColorSpan = new BackgroundColorSpan(context.getResources().getColor(R.color.colorAccent));
        spannableString.setSpan(backgroundColorSpan,3,5,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }


    public static SpannableString getStrikethroughSpan(){
        SpannableString spannableString = new SpannableString("尤龙的传人");
        StrikethroughSpan strikethroughSpan = new StrikethroughSpan();
        spannableString.setSpan(strikethroughSpan,0,1,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

    public static SpannableString getUnderlineSpan(){
        SpannableString spannableString = new SpannableString("这里是下划线");
        UnderlineSpan underlineSpan = new UnderlineSpan();
        spannableString.setSpan(underlineSpan,3,6,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }

    public static SpannableString getScaleXSpan(){
        SpannableString spannableString = new SpannableString("媳妇你长胖了");
        ScaleXSpan scaleXSpan = new ScaleXSpan(2);
        spannableString.setSpan(scaleXSpan,0,spannableString.length(),Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }

    public static SpannableString getSuperscriptSpan(){
        SpannableString spannableString = new SpannableString("刚在北京望京买了套1202m的房子");
        SuperscriptSpan superscriptSpan = new SuperscriptSpan();
        RelativeSizeSpan sizeSpan = new RelativeSizeSpan(0.7f);
        spannableString.setSpan(superscriptSpan,12,13,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan,12,13,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

    public static SpannableString getSubscriptSpan(){
        SpannableString spannableString = new SpannableString("水分子化学式为H20");
        SubscriptSpan subscriptSpan = new SubscriptSpan();
        RelativeSizeSpan sizeSpan = new RelativeSizeSpan(0.7f);
        spannableString.setSpan(subscriptSpan,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

    public static SpannableString getStyleSpan(){
        SpannableString spannableString = new SpannableString("身正不怕影子歪");
        StyleSpan styleSpanBold = new StyleSpan(Typeface.BOLD);
        StyleSpan styleSpanitalic = new StyleSpan(Typeface.ITALIC);
        spannableString.setSpan(styleSpanBold,0,4,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        spannableString.setSpan(styleSpanitalic,4,6,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

    public static SpannableString getRelativeSizeSpan(){
        SpannableString spannableString = new SpannableString("我心情忐忑不安七上八下");
        RelativeSizeSpan sizeSpan1 = new RelativeSizeSpan(1.2f);
        RelativeSizeSpan sizeSpan2 = new RelativeSizeSpan(1.4f);
        RelativeSizeSpan sizeSpan3 = new RelativeSizeSpan(1.6f);
        RelativeSizeSpan sizeSpan4 = new RelativeSizeSpan(1.8f);

        spannableString.setSpan(sizeSpan1,0,1,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,1,2,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,2,3,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan4,3,4,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,4,5,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,5,6,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan1,6,7,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan2,7,8,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan3,8,9,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(sizeSpan4,9,10,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);

        return spannableString;
    }

    public static SpannableString getImageSpan(Context context){
        SpannableString spannableString = new SpannableString("芭芭拉小魔仙 魔法棒");
        Drawable drawable = context.getResources().getDrawable(R.mipmap.fx);
        drawable.setBounds(0,0,70,70);
        ImageSpan imageSpan = new ImageSpan(drawable);
        spannableString.setSpan(imageSpan,6,7,Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

    public static SpannableString getClickableSpan(final Context context){
        SpannableString spannableString = new SpannableString("哪里不会点哪里,so easy!");
        ClickableSpan clickableSpan = new ClickableSpan() {
            @Override
            public void onClick(View widget) {
                Toast.makeText(context,"你戳中我了",Toast.LENGTH_SHORT).show();
            }
        };
        spannableString.setSpan(clickableSpan,5,7,Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        return spannableString;
    }

    public static SpannableString getURLSpan(){
        SpannableString spannableString = new SpannableString("打电话,发短信,发邮件,打开网页");
        spannableString.setSpan(new URLSpan("tel:10086"), 0, 3, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("smsto:10086"), 4, 7, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("mailto:[email protected]"), 8, 11, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new URLSpan("http://www.jianshu.com"), 12, 16, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        return spannableString;
    }

}

MainActivity完整代码:

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.text.method.LinkMovementMethod;
import android.widget.TextView;
import butterknife.Bind;
import butterknife.ButterKnife;

public class MainActivity extends AppCompatActivity {

    @Bind(R.id.tv_foreground)
    TextView tvForeground;
    @Bind(R.id.tv_background)
    TextView tvBackground;
    @Bind(R.id.tv_relativesize)
    TextView tvRelativesize;
    @Bind(R.id.tv_strikethrough)
    TextView tvStrikethrough;
    @Bind(R.id.tv_underline)
    TextView tvUnderline;
    @Bind(R.id.tv_superscript)
    TextView tvSuperscript;
    @Bind(R.id.tv_subscript)
    TextView tvSubscript;
    @Bind(R.id.tv_style)
    TextView tvStyle;
    @Bind(R.id.tv_image)
    TextView tvImage;
    @Bind(R.id.tv_click)
    TextView tvClick;
    @Bind(R.id.tv_url)
    TextView tvUrl;
    @Bind(R.id.tv_scalex)
    TextView tvScalex;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);

        tvForeground.setText(SpanUtils.getForegroundColorSpan(getApplicationContext()));

        tvBackground.setText(SpanUtils.getBackgroundColorSpan(getApplicationContext()));

        tvStrikethrough.setText(SpanUtils.getStrikethroughSpan());

        tvUnderline.setText(SpanUtils.getUnderlineSpan());

        tvScalex.setText(SpanUtils.getScaleXSpan());

        tvSuperscript.setText(SpanUtils.getSuperscriptSpan());

        tvSubscript.setText(SpanUtils.getSubscriptSpan());

        tvStyle.setText(SpanUtils.getStyleSpan());

        tvRelativesize.setText(SpanUtils.getRelativeSizeSpan());

        tvImage.setText(SpanUtils.getImageSpan(getApplicationContext()));

        tvClick.setMovementMethod(LinkMovementMethod.getInstance());  //否则点击没反应
        tvClick.setText(SpanUtils.getClickableSpan(getApplicationContext()));

        tvUrl.setMovementMethod(LinkMovementMethod.getInstance());
        tvUrl.setText(SpanUtils.getURLSpan());
    }
}

文章只是讲解单一,组合拳威力更大,系列下一篇会讲自定义Span开发更炫酷的效果。

你可能感兴趣的:(给TextView插上SpannableString的翅膀)