富文本 SpannableString Span
富文本类简介
设置富文本信息的几种方式:
- 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;@Overrideprotected 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" ><TextViewandroid:id="@+id/tv1"android:layout_width="wrap_content"android:layout_height="wrap_content" /><TextViewandroid:id="@+id/tv2"android:layout_width="wrap_content"android:layout_height="wrap_content" /><TextViewandroid:id="@+id/tv3"android:layout_width="wrap_content"android:layout_height="wrap_content" /><TextViewandroid:id="@+id/tv4"android:layout_width="wrap_content"android:layout_height="wrap_content" /><TextViewandroid: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的更多相关文章
- Vue富文本编辑器(图片拖拽缩放)
富文本编辑器(图片拖拽缩放) 需求: 根据业务要求,需要能够上传图片,且上传的图片能在移动端中占满屏幕宽度,故需要能等比缩放上传的图片,还需要能拖拽.缩放.改变图片大小.尝试多个第三方富文本编辑器,很 ...
- springmvc 后台实体类接受前端json字符串时,其中一个属性content 接受富文本内容时 标签<p>、<span> 这些标签丢失问题解决
问题描述: 前端一个字段 <script id="editor" type="text/plain" name="content" s ...
- SpannableString富文本
忍不住想吐槽这个类,这个类是要给文本设置不同的颜色.字体样式 例子:一句话中只有某几个文字想要设置成不同的颜色 起初写了一个函数setColorStyle(), public SpannableStr ...
- TextView展示富文本时emoj或图片和文字不对齐的解决方案
在项目中,回复框.聊天界面的显示往往会有emoj或者图片,但是一个比较头疼的问题是,会出现emoj表情或者图片和文字的位置不对齐,总是有偏移,这样很影响用户体验的.下面会总结一下如何解决这个问题. 本 ...
- Android - 富文本编辑器
Android富文本编辑器(一):基础知识 目前主流的基于Android富文本开发方式思路如下: 基于TextView图文混排 使用方式: TextView textView = new TextVi ...
- 个人网站对xss跨站脚本攻击(重点是富文本编辑器情况)和sql注入攻击的防范
昨天本博客受到了xss跨站脚本注入攻击,3分钟攻陷--其实攻击者进攻的手法很简单,没啥技术含量.只能感叹自己之前竟然完全没防范. 这是数据库里留下的一些记录.最后那人弄了一个无限循环弹出框的脚本,估计 ...
- 富文本编辑器kindeditor配置
<!--富文本编辑器kindeditor配置↓ --> <link type="text/css" rel="stylesheet" href ...
- 富文本编辑器防止xss注入javascript版
富文本编辑器:ueditor 其实富文本编辑器已经有防止xss注入功能,但是你服务端程序在接收的时候在做一次转义,否则有可能然后前端验证直接提交数据导致被xss攻击. 为了节省后端程序开销则在前端 显 ...
- 对于MVC中应用百度富文本编辑器问题的解决办法
1.对于应用富文本编辑器post提交表单内容提示有危险的解决办法: [ValidateInput(false)] //文本编辑器的表单提交不用提示危险 [HttpPost] public Action ...
随机推荐
- [转自已]Windos多个文件快速重命名说明+图解
转自己以前的文章,给新博客带点气氛. 1.(复制的)比如在文件夹中包含yin.jpg.ye.jpg.zou.jpg三个文件,你希望将它们命名为"photo+数字"的文件名形式,那么 ...
- 在IE6/7/8下识别html5标签
识别html5标签: html5添加了许多语义化的标签,比如<nav></nav>,<aside></aside>,<article>< ...
- python之路基础篇
基础篇 1.Python基础之初识python 2.Python数据类型之字符串 3.Python数据类型之列表 4.Python数据类型之元祖 5.Python数据类型之字典 6.Python Se ...
- python中的几种遍历列表的方法比较
python的内容非常丰富,给我们带来的便利很多,很多事情的表达方法有很大的多样性,比如我经常需要遍历一个列表,取它的下标和值,这个时候就有很多方法需要取舍一下才行. for循环遍历 l = [1,2 ...
- UVA - 12627 Erratic Expansion 奇怪的气球膨胀 (分治)
紫书例题p245 Piotr found a magical box in heaven. Its magic power is that if you place any red balloon i ...
- UVA - 524 Prime Ring Problem(dfs回溯法)
UVA - 524 Prime Ring Problem Time Limit:3000MS Memory Limit:0KB 64bit IO Format:%lld & % ...
- Java中堆内存(heap)和栈内存(stack)的区别
在Java代码中,常常会使用到这样的类的声明实例化: Person per = new Person(); //这其实是包含了两个步骤,声明和实例化 Person per = null; //声明一个 ...
- asp.net 后台使用js弹窗失效问题
1.这些事件输出来前后都变成JS代码了,看到到这样的代码的了.会变成<script>alert('合同号XXX已存在')</script>首先后台调试一下看看Page.Clie ...
- 在vs中连接sql的几种连接方式
sql身份验证:Data Source=.;Initial Catalog=DBName;UID=sa;Pwd=pwd windows身份验证:Data Source=.;Initial Catal ...
- Windows系统编程之进程间通信
Windows系统编程之进程间通信作者:北极星2003来源:看雪论坛(www.pediy.com)Windows 的IPC(进程间通信)机制主要是异步管道和命名管道.(至于其他的IPC方式,例如内存映 ...