Android 富文本SpannableString

news2025/1/4 19:06:46

一、认识SpannableString

  • 为什么要使用富文本
    在Android开发中,有很多UI会画出一些特别炫酷的界面出来,比如一个字符串里有特殊的字会有其他颜色并加粗、变大变小、插入小图片、给某几个文字添加边框,如果我们使用笨办法用几个TextView或者ImageView来链接,这样虽然能实现但是会不会很笨重,如果出现换行就尴尬了,他不能想做到无缝换行造成效果跟预期效果相差太大,如果效果很复杂是不是会给应用增加体验负担?还会给带来适配的问题,SpannableString的出现帮我们解决了这一系列的问题。

  • 他能给我们带来什么
    解决复杂的字符串效果UI效果,

二、 如何实现SpannableString的功能效果

  • 比较常用的Span样式
  1. BackgroundColorSpan : 文本背景色、
  2. ForegroundColorSpan : 文本颜色
  3. MaskFilterSpan : 修饰效果,如模糊(BlurMaskFilter)浮雕
  4. RasterizerSpan : 光栅效果
  5. StrikethroughSpan : 删除线
  6. SuggestionSpan : 相当于占位符
  7. UnderlineSpan : 下划线
  8. AbsoluteSizeSpan : 文本字体(绝对大小)
  9. DynamicDrawableSpan : 设置图片,基于文本基线或底部对齐。
  10. ImageSpan : 图片
  11. RelativeSizeSpan : 相对大小(文本字体)
  12. ScaleXSpan : 基于x轴缩放
  13. StyleSpan : 字体样式:粗体、斜体等
  14. SubscriptSpan : 下标(数学公式会用到)
  15. SuperscriptSpan : 上标(数学公式会用到)
  16. TextAppearanceSpan : 文本外貌(包括字体、大小、样式和颜色)
  17. TypefaceSpan : 文本字体
  18. URLSpan : 文本超链接
  19. ClickableSpan : 点击事件

效果:

在这里插入图片描述

public class CommShowSpanActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_comm_show_span);
        TextView text = findViewById(R.id.text);
        text.setTextSize(23);

        String str = "东风路看对方了困难发了多少你疯啦百度一下上岛咖啡了上来看待离开克里斯丁肺宁颗粒道德积分呢小四的减肥我欧艾斯都放开克里斯丁肺宁颗粒道德积分呢小四的减肥我欧艾斯都放开克里斯丁肺宁颗粒道德积分呢小四的减肥我欧艾斯都放假哦吉安市都放假哦so";
        SpannableString spannableString = new SpannableString(str);


        /*字体变大变小*/
        AbsoluteSizeSpan sizeSpan = new AbsoluteSizeSpan(80);
        spannableString.setSpan(sizeSpan, 0, 3, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*某个字符串改变颜色*/
        ForegroundColorSpan foregroundColorSpan = new ForegroundColorSpan(Color.RED);
        spannableString.setSpan(foregroundColorSpan, 3, 6, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*字符串的背景色*/
        BackgroundColorSpan backgroundColorSpan = new BackgroundColorSpan(Color.YELLOW);
        spannableString.setSpan(backgroundColorSpan, 6, 8, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*下划线  颜色随字体颜色 */
        UnderlineSpan underlineSpan = new UnderlineSpan();
        spannableString.setSpan(underlineSpan, 0, 8, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*删除线*/
        StrikethroughSpan strikethroughSpan = new StrikethroughSpan();
        spannableString.setSpan(strikethroughSpan, 8, 12, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);

        /*暂位符*/
        SuggestionSpan suggestionSpan = new SuggestionSpan(CommShowSpanActivity.this, new String[]{
                "0000", "1111"
        }, SuggestionSpan.FLAG_EASY_CORRECT);
        spannableString.setSpan(suggestionSpan, 8, 12, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        //Typeface  加粗
        StyleSpan styleSpan = new StyleSpan(Typeface.BOLD);
        spannableString.setSpan(styleSpan, 12, 16, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*斜体文本*/
        StyleSpan styleSpan1 = new StyleSpan(Typeface.ITALIC);
        spannableString.setSpan(styleSpan1, 12, 16, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*超链接*/
        URLSpan urlSpan = new URLSpan("http://www.baidu.com");
        spannableString.setSpan(urlSpan, 16, 20, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /*文本字体*/
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.P) {
            TypefaceSpan typefaceSpan = new TypefaceSpan("");
            spannableString.setSpan(typefaceSpan, 16, 20, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        }


        /*修饰效果,如模糊(BlurMaskFilter)*/
        MaskFilterSpan maskFilterSpan = new MaskFilterSpan(new BlurMaskFilter(12, BlurMaskFilter.Blur.SOLID));
        spannableString.setSpan(maskFilterSpan, 20, 25, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
        //浮雕
        MaskFilterSpan maskFilterSpan1 = new MaskFilterSpan(new EmbossMaskFilter(new float[]{
                10, 20, 30
        }, 0.5f, 0.5f, 15));
        spannableString.setSpan(maskFilterSpan1, 25, 30, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);



        //TextAppearanceSpan 文本外貌(包括字体、大小、样式和颜色)
        TextAppearanceSpan textAppearanceSpan = new TextAppearanceSpan(this, R.style.text);
        spannableString.setSpan(textAppearanceSpan, 30, 33, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        TextAppearanceSpan textAppearanceSpan1 = new TextAppearanceSpan(this,
                android.R.style.TextAppearance_Theme,getResources().getColor(R.color.colorPrimary));
        spannableString.setSpan(textAppearanceSpan1, 34, 35, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);


        /**
         * DynamicDrawableSpan 设置图片,基于文本基线或底部对齐。
         */
        DynamicDrawableSpan drawableSpan =
                new DynamicDrawableSpan(DynamicDrawableSpan.ALIGN_BASELINE) {
                    @Override
                    public Drawable getDrawable() {
                        Drawable d = getResources().getDrawable(R.mipmap.add_pic);
                        d.setBounds(0, 0, d.getIntrinsicWidth()/2, d.getIntrinsicHeight()/2);
                        return d;
                    }
                };
        spannableString.setSpan(drawableSpan, 3, 4, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);




        DynamicDrawableSpan drawableSpan2 = new DynamicDrawableSpan(
                DynamicDrawableSpan.ALIGN_BOTTOM) {
            @Override
            public Drawable getDrawable() {
                Drawable d = getResources().getDrawable(R.mipmap.add_pic);
                d.setBounds(0, 0, 50, 50);
                return d;
            }
        };
        spannableString.setSpan(drawableSpan2, 7, 8, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);

        /**
         * 设置字体相对大小
         */
        RelativeSizeSpan relativeSizeSpan = new RelativeSizeSpan(2);
        spannableString.setSpan(relativeSizeSpan, 50, 60, Spannable.SPAN_INCLUSIVE_INCLUSIVE);


        /**
         * 下标脚注
         */
        SubscriptSpan subscriptSpan = new SubscriptSpan();
        spannableString.setSpan(subscriptSpan, 69, 70, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new ForegroundColorSpan(Color.RED),69, 70, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        /**
         * 上标
         */
        SuperscriptSpan superscriptSpan = new SuperscriptSpan();
        spannableString.setSpan(superscriptSpan, 76, 77, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        spannableString.setSpan(new ForegroundColorSpan(Color.RED),76, 77, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);

        text.setText(spannableString);
        text.setMovementMethod(LinkMovementMethod.getInstance());
    }
}

  • ReplacementSpan给一串字符串中的某两个文字添加背景或者边框

效果:
在这里插入图片描述

代码:

public class TagBgSpan extends ReplacementSpan {

    /**
     * 当前tag的宽度
     */
    public int currentMeasureTextWidth;
    /**
     * 左右間距
     */
    public int paddingLandR = 10;
    /**
     * 标签之前的间距
     */
    public int speed = 10;
    /**
     * 圆角
     */
    private int radius = 10;

    /**
     * 背景色
     */
    private int bgColor = Color.GRAY;

    /**
     * 画笔的风格
     */
    private Paint.Style style = Paint.Style.STROKE;

    /**
     * 標簽颜色
     */
    private int tagTextColor = Color.RED;


    public TagBgSpan setTagTextColor(int tagTextColor) {
        this.tagTextColor = tagTextColor;
        return this ;
    }

    public TagBgSpan setStyle(Paint.Style style) {
        this.style = style;
        return this ;
    }


    public TagBgSpan setBgColor(int bgColor) {
        this.bgColor = bgColor;
        return this;
    }


    public TagBgSpan setSpeed(int speed) {
        this.speed = speed;
        return this;
    }

    public TagBgSpan setRadius(int radius) {
        this.radius = radius;
        return this ;
    }


    @Override
    public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) {
        currentMeasureTextWidth = (int) (paint.measureText(text.toString(), start, end) + (paddingLandR * 2) + (speed * 2));
        return currentMeasureTextWidth;
    }


    @Override
    public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) {
        paint.setColor(bgColor);
        paint.setStyle(style);
        RectF rect = new RectF(x + speed, top, x + currentMeasureTextWidth - speed, bottom);
        canvas.drawRoundRect(rect, radius, radius, paint);


        paint.setColor(tagTextColor);
        paint.setStyle(Paint.Style.FILL_AND_STROKE);
        canvas.drawText(text, start, end, x + speed + paddingLandR, y, paint);
    }
}

  • ClickableSpan实现一串字符串中某几个文字支持点击功能
    效果:
    效果

代码:

public class MyClickableSpan extends ClickableSpan {

    private Context context ;

    public MyClickableSpan(Context context ){
        this.context=context;
    }

    @Override
    public void updateDrawState(@NonNull TextPaint ds) {
        super.updateDrawState(ds);
        ds.setColor(Color.RED);
        ds.setFakeBoldText(true);
    }

    @Override
    public void onClick(@NonNull View widget) {
        Toast.makeText(context,"测试onClick",Toast.LENGTH_SHORT).show();
    }
}

  • ImageSpan实现在字符串中插入图片

效果:
在这里插入图片描述

代码:

public class MyImageSpan extends ImageSpan {


    public MyImageSpan(@NonNull Context context, @NonNull Bitmap bitmap) {
        super(context, bitmap);
    }


    @Override
    public int getSize(@NonNull Paint paint, CharSequence text, int start, int end, @Nullable Paint.FontMetricsInt fm) {
        return getDrawable().getIntrinsicWidth();
    }


    @Override
    public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint) {
        super.draw(canvas, text, start, end, x, top, y, bottom, paint);
        Drawable b = getDrawable();
        Paint.FontMetricsInt fm = paint.getFontMetricsInt();
        int transY = (y + fm.descent + y + fm.ascent) / 2 - b.getBounds().bottom / 2;
        canvas.save();
        canvas.translate(x, transY);
        b.draw(canvas);
        canvas.restore();
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解决Spring Boot 2.7.16 在服务器显示启动成功无法访问问题:从本地到服务器的部署坑

🌷🍁 博主猫头虎 带您 Go to New World.✨🍁 🦄 博客首页——猫头虎的博客🎐 🐳《面试题大全专栏》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺 &a…

JavaWeb 学习笔记 8:AJAX

JavaWeb 学习笔记 8:AJAX AJAX(Asynchronous JavaScript And XML,异步 js 和 XML)是一种用 js 代码异步(或同步)的方式请求服务端数据,并在页面显示或加载的技术。 1.快速入门 先看如何用纯 …

GB28181协议-注册详解

注册指的是设备或系统进入联网系统时向SIP服务器(SIP UAS)进行注册登记的工作模式,在本文中FFmpeg即为一个SIP服务器,设备向FFmpeg发送注册请求,FFmpeg在接收到设备的注册请求后返回相应的回复消息,则完成设…

【面试题】——Spring

1.Spring是什么? Spring是一个开源的Java应用框架,它提供了广泛的基础设施支持,用于构建Java应用程序。极大提高了开发效率。它提供了一种轻量级的编程模型,通过依赖注入(Dependency Injection)和面向切面…

盘点神级AI绘画工具,太强太友好了

B站|公众号:啥都会一点的研究生 AI绘画越来越卷、越来越极致,可以轻松将我们这些毫无绘画功底的人的创意想法转化为令人惊叹的数字艺术作品,本期将整理一些目前流行的高质量AI艺术生成器,一起看看吧 Bing Image Creator https:/…

污水中常规五参数的检测有什么意义

便携式水质五参数检测仪包括: 温度、 pH、溶解氧、电导率、浊度。同时该仪器也可根据客户需要进行其他电极法指标, 如盐度、 TD S、OR P 指标的增添,以及指标的任意搭配组合。 水质常规五参数在我国水污染防治中发挥着重要的作用。 检测水质常…

spring中AbstractApplicationContext的refresh()

这个部分的源码看的次数不计其数了,每次看不同开源组件的调用过程中走到这里都有不同的收获。 spring 中 AbstractApplicationContext 的 refresh() 是 spring 的核心,几乎所有的逻辑都在是这里间接被调用。 如下源码为 spring boot 2.7.15 自带的 sprin…

MySQL主从复制与读写分离(附配置实例)

目录 一、主从复制1.1 什么是主从复制?1.2 MySQL支持的复制类型1.3 主从复制的工作过程1.4 主从复制的同步模式1.4.1 异步复制(Asynchronous replication)1.4.2 全同步复制(Fully synchronous replication)1.4.3 半同步…

其他网页都可以打开,只有知网打不开

问题:电脑自动获取DNS出现问题。 1、打开Internet协议版本4(TCP/IPv4) 控制面板—所有控制面板项----网络连接----WLAN—属性----Internet协议版本4(TCP/IPv4)—属性 2、修改DNS 将自动获得DNS服务器地址&#xff…

Sparta工具用法描述之信息收集(漏洞分析)

声明:本文仅做学习与交流,任何用于非法用途、行为等造成他人损失的,责任自负。本文不承担任何法律责任。 Sparta是python GUI应用程序,它通过在扫描和枚举阶段协助渗透测试仪来简化网络基础结构渗透测试。 通过点击并单击工具箱并以方便的方式显示所有工具输出,它可以使测…

Swing通过后台线程实现页面更新

业务场景 在swing程序中,我们想实现一个随着任务的执行而同步更新组件的功能,如果在主线程(EDT)中直接执行任务并更新组件,则会导致组件只显示最终更新的状态的问题,这是因为EDT是单线程的,在执…

【golang】深入理解Go语言垃圾回收(GC)

垃圾回收 垃圾回收版本1.3之前标记-清除(mark and sweep)算法标记-清除(mark and sweep)的缺点 版本1.5的三色并发标记法没有STW的三色标记法屏障机制强-弱 三色不等式插入屏障删除屏障 版本1.8的混合写屏障(hybrid wr…

MQ - 25 RabbitMQ集群架构设计与实现

文章目录 导图集群构建节点发现元数据存储数据可靠性镜像队列仲裁队列安全控制传输加密身份认证资源鉴权可观测性总结导图 集群构建 集群构建由节点发现和元数据存储两部分组成。RabbitMQ 也是一样的实现思路。 节点发现 在节点发现方面,RabbitMQ 通过插件化的方式支持了多…

经典题记录 字符串相加/相乘

1. LeetCode 415 字符串相加 代码一:代码简短,但需要借助额外的一个string来保存结果,更占用内存。 class Solution { public:string addStrings(string num1, string num2) {string ans"";int size1num1.size();int size2num2.si…

php之导入导出csv文件

一、导入csv文件 1、创建导入页面 <!DOCTYPE html> <html lang"zh-CN"> <meta charset"UTF-8"> <head><title>文件提交表单</title> </head> <body> <form action"test5.php" method"…

嵌入式裸机架构的探索与崩塌

为什么会想着探索下嵌入式裸机的架构呢&#xff1f;是因为最近写了一个项目&#xff0c;项目开发接近尾声时&#xff0c;发现了一些问题&#xff1a; 1、项目中&#xff0c;驱动层和应用层掺杂在一起&#xff0c;虽然大部分是应用层调用驱动层&#xff0c;但是也存在驱动层调用…

6年Android开发前10月的总结,写给正在求职的安卓开发

进入大厂工作对许多人来说已经是一种挑战&#xff0c;但只要充分准备&#xff0c;很多问题都可以逐步解决。当然&#xff0c;运气也起到了一定的作用&#xff0c;但最终还是与自身的努力密不可分。运气是实力的一部分&#xff0c;因为自助者天助。 每到10月进行总结时&#xff…

华为ICT——第二章-数字图像处理私人笔记

目录 1&#xff1a;计算机视觉&#xff1a;​编辑 2&#xff1a;计算机视觉应用&#xff1a;​编辑 3&#xff1a;计算机视界核心问题&#xff1a;​编辑 4&#xff1a;相关学科&#xff1a; 5&#xff1a;计算机视觉与人工智能&#xff1a; 最成熟的技术方向是图像识别 6…

Django应用及分布式路由

Django应用及分布式路由 应用 应用在Django项目中一个完全独立的业务模块&#xff0c;可以包含自己的路由&#xff0c;视图&#xff0c;模板&#xff0c;模型 应用配置 在这里面添加你自定义的应用 INSTALLED_APPS [django.contrib.admin,django.contrib.auth,django.cont…

如何使用docker快速部署MinDoc文档系统

MinDoc是非常优秀的知识分享系统&#xff0c;但是很多刚接触的人会一脸懵逼&#xff0c;而且官方文档写的也并不清晰&#xff0c;所以和大家分享一下快速部署MinDoc的方法。 首先docker环境先自行安装好&#xff0c;这里不再赘述。 拉取docker镜像&#xff1a; docker pull …