SpannableString其實和String一樣,都是一種字元串類型,SpannableString可以直接作為TextView的顯示文本,不同的是SpannableString可以通過使用其方法setSpan方法實作字元串各種形式風格的顯示,重要的是可以指定設定的區間,也就是為字元串指定下标區間内的子字元串設定格式。
setSpan(Object what, int start, int end, int flags)方法需要使用者輸入四個參數,
what
表示設定的格式是什麼,可以是前景色、背景色也可以是可點選的文本等等,
start
表示需要設定格式的子字元串的起始下标,同理
end
表示終了下标,
flags
屬性就有意思了,共有四種屬性:
Spanned.SPAN_INCLUSIVE_EXCLUSIVE
從起始下标到終了下标,包括起始下标
Spanned.SPAN_INCLUSIVE_INCLUSIVE
從起始下标到終了下标,同時包括起始下标和終了下标
Spanned.SPAN_EXCLUSIVE_EXCLUSIVE
從起始下标到終了下标,但都不包括起始下标和終了下标
Spanned.SPAN_EXCLUSIVE_INCLUSIVE
從起始下标到終了下标,包括終了下标
SpannableString的setSpan()方法可以同時使用多個,實作多種效果疊加。
下面我們一一解讀幾種Span常用的格式:
- ForegroundColorSpan

ForegroundColorSpan
,為文本設定前景色,效果和TextView的setTextColor()類似,實作方法如下:
SpannableString spannableString = new SpannableString("設定文字的前景色為淡藍色");
ForegroundColorSpan colorSpan = new ForegroundColorSpan(Color.parseColor("#0099EE"));
spannableString.setSpan(colorSpan, 9, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
設定的區間是9到字元串的最後,也就是圖中“淡藍色”三字。
- BackgroundColorSpan
BackgroundColorSpan
,為文本設定背景色,效果和TextView的setBackground()類,實作方法如下:
SpannableString spannableString = new SpannableString("設定文字的背景色為淡綠色");
BackgroundColorSpan colorSpan = new BackgroundColorSpan(Color.parseColor("#AC00FF30"));
spannableString.setSpan(colorSpan, 9, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
- RelativeSizeSpan
RelativeSizeSpan
,設定文字相對大小,在TextView原有的文字大小的基礎上,相對設定文字大小,實作方法如下:
SpannableString spannableString = new SpannableString("萬丈高樓平地起");
RelativeSizeSpan sizeSpan01 = new RelativeSizeSpan(1.2f);
RelativeSizeSpan sizeSpan02 = new RelativeSizeSpan(1.4f);
RelativeSizeSpan sizeSpan03 = new RelativeSizeSpan(1.6f);
RelativeSizeSpan sizeSpan04 = new RelativeSizeSpan(1.8f);
RelativeSizeSpan sizeSpan05 = new RelativeSizeSpan(1.6f);
RelativeSizeSpan sizeSpan06 = new RelativeSizeSpan(1.4f);
RelativeSizeSpan sizeSpan07 = new RelativeSizeSpan(1.2f);
spannableString.setSpan(sizeSpan01, 0, 1, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan02, 1, 2, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan03, 2, 3, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan04, 3, 4, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan05, 4, 5, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan06, 5, 6, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(sizeSpan07, 6, 7, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
- StrikethroughSpan
StrikethroughSpan
,為文本設定中劃線,也就是常說的删除線,實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定删除線");
StrikethroughSpan strikethroughSpan = new StrikethroughSpan();
spannableString.setSpan(strikethroughSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
看到這有沒有小激動,分分鐘實作天貓打折優惠效果,有木有?
- UnderlineSpan
UnderlineSpan
,為文本設定下劃線,具體實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定下劃線");
UnderlineSpan underlineSpan = new UnderlineSpan();
spannableString.setSpan(underlineSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
- SuperscriptSpan
SuperscriptSpan
,設定上标,具體實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定上标");
SuperscriptSpan superscriptSpan = new SuperscriptSpan();
spannableString.setSpan(superscriptSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
從效果圖可以看出,被設定為上标的文字大小和下面的文本文字大小一樣,隻要我們稍加修飾,結合
RelativeSizeSpan
設定小字型文本作為上标,分分鐘實作指數公式有木有,再也不用
2^2+3^2=13
這樣缺乏審美的數學公式了,是不是超實用?
- SubscriptSpan
SubscriptSpan
,設定下标,功能與設定上标類似,不做過多描述,具體實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定下标");
SubscriptSpan subscriptSpan = new SubscriptSpan();
spannableString.setSpan(subscriptSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
- StyleSpan
StyleSpan
,為文字設定風格(粗體、斜體),和TextView屬性textStyle類似,實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定粗體、斜體風格");
StyleSpan styleSpan_B = new StyleSpan(Typeface.BOLD);
StyleSpan styleSpan_I = new StyleSpan(Typeface.ITALIC);
spannableString.setSpan(styleSpan_B, 5, 7, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
spannableString.setSpan(styleSpan_I, 8, 10, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setHighlightColor(Color.parseColor("#36969696"));
textView.setText(spannableString);
- ImageSpan
ImageSpan
,設定文本圖檔,實作方法如下:
SpannableString spannableString = new SpannableString("在文本中添加表情(表情)");
Drawable drawable = getResources().getDrawable(R.mipmap.a9c);
drawable.setBounds(0, 0, 42, 42);
ImageSpan imageSpan = new ImageSpan(drawable);
spannableString.setSpan(imageSpan, 6, 8, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setText(spannableString);
這一個是不是很炫酷?再加一個解析算法,将文本中特定的文本轉換成特定的表情圖檔,分分鐘實作聊天表情顯示效果有木有啊朋友們!
- ClickableSpan
ClickableSpan
,設定可點選的文本,設定這個屬性的文本可以相應使用者點選事件,至于點選事件使用者可以自定義,就像效果圖顯示一樣,使用者可以實作點選跳轉頁面的效果,具體實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定點選事件");
MyClickableSpan clickableSpan = new MyClickableSpan("http://www.jianshu.com/users/dbae9ac95c78");
spannableString.setSpan(clickableSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setHighlightColor(Color.parseColor("#36969696"));
textView.setText(spannableString);
/***************************************************************/
class MyClickableSpan extends ClickableSpan {
private String content;
public MyClickableSpan(String content) {
this.content = content;
}
@Override
public void updateDrawState(TextPaint ds) {
ds.setUnderlineText(false);
}
@Override
public void onClick(View widget) {
Intent intent = new Intent(MainActivity.this, OtherActivity.class);
Bundle bundle = new Bundle();
bundle.putString("content", content);
intent.putExtra("bundle", bundle);
startActivity(intent);
}
}
代碼中我們自定義MyClickableSpan類,繼承至ClickableSpan,并重寫其中一些方法。ds.setUnderlineText()控制是否讓可點選文本顯示下劃線,很明顯,在上面代碼中我選擇了false,不顯示下滑寫。onClick點選事件的具體實作方法寫在其中。如上代碼,我們重寫ClickableSpan的onClick方法實作Activity的跳轉效果,并傳遞跳轉資料。
注意:使用ClickableSpan的文本如果想真正實作點選作用,必須為TextView設定setMovementMethod方法,否則沒有點選相應,至于setHighlightColor方法則是控制點選是的背景色。
- URLSpan
URLSpan
,設定超連結文本,其實聰明的小夥幫在講到
ClickableSpan
的時候就能實作超連結文本的效果了,重寫onClick點選事件就行,也确實看了
URLSpan
的源碼,
URLSpan
就是繼承自
ClickableSpan
,也和想象中一樣,就是重寫了父類的onClick事件,用系統自帶浏覽器打開連結,具體實作方法如下:
SpannableString spannableString = new SpannableString("為文字設定超連結");
URLSpan urlSpan = new URLSpan("http://www.jianshu.com/users/dbae9ac95c78");
spannableString.setSpan(urlSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setHighlightColor(Color.parseColor("#36969696"));
textView.setText(spannableString);
URLSpan
onClick事件的源碼如下:
@Override
public void onClick(View widget) {
Uri uri = Uri.parse(getURL());
Context context = widget.getContext();
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
intent.putExtra(Browser.EXTRA_APPLICATION_ID, context.getPackageName());
try {
context.startActivity(intent);
} catch (ActivityNotFoundException e) {
Log.w("URLSpan", "Actvity was not found for intent, " + intent.toString());
}
}
除此之外,還有
MaskFilterSpan
可以實作模糊和浮雕效果,
RasterizerSpan
可以實作光栅效果,因為以上兩個使用頻率不高,而且效果也不是很明顯,就不做詳細說明,有興趣的小夥伴不妨去試一試。
SpannableStringBuilder
應該有不少開發的小夥伴知道StringBuilder,可以使用append()方法實作字元串拼接,非常友善。同樣,SpannableString中也有SpannableStringBuilder,顧名思義,就是實作對,SpannableString的一個拼接效果,同樣是append()方法,可以實作各種風格效果的SpannableString拼接,非常實用。
作者:碼農小阿飛CoderMario
連結:http://www.jianshu.com/p/84067ad289d2
來源:簡書
著作權歸作者所有。商業轉載請聯系作者獲得授權,非商業轉載請注明出處。