TeXtView你們應該都不陌生,文本展現控件嘛! 就用TextView顯示普普統統的文本,OK,很簡單,Android入門的都會,沒入門的在門縫外看兩眼也都會,哈哈,開玩笑。那要是設計在開發需求中要求相似微信聊天表情同樣在TextView中插入表情圖片呢? 有的小夥伴就會說啦,「TextView添加圖片我會啊,不就是drawableLeft
,drawableRight
嘛!」 嗯 ~ 也行,算是一種方法,可這有一個限制,首先,圖片只能在TextView的兩端,其次,兩端都只能設置一張圖片。要是圖片要在文本中間呢?無能爲力了吧,要是你了會使用SpannableString,這個問題也就不難解決了,簡直是Just So So。git
因此,不論你是否正在經受以上問題的困擾,亦或是尚未經歷到,請駐足仔細耐心的看完這篇簡短的文章。不只可以輕鬆實現以上設計需求,更能收穫其餘各類炫酷的效果,也許就能幫助你解決如今你所困擾的問題。github
首先咱們先來了解SpannableString算法
SpannableString其實和String同樣,都是一種字符串類型,一樣TextView也能夠直接設置SpannableString做爲顯示文本,不一樣的是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
從起始下標到終了下標,包括終了下標app
下面咱們一一解讀幾種Span經常使用的格式:ide
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到字符串的最後,也就是圖中「淡藍色」三字。ui
BackgroundColorSpan
,爲文本設置背景色,效果和TextView的setBackground()類,實現方法以下:this
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
,設置文字相對大小,在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
,爲文本設置中劃線,也就是常說的刪除線,實現方法以下:
SpannableString spannableString = new SpannableString("爲文字設置刪除線"); StrikethroughSpan strikethroughSpan = new StrikethroughSpan(); spannableString.setSpan(strikethroughSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE); textView.setText(spannableString);
看到這有沒有小激動,分分鐘實現天貓打折優惠效果,有木有?
UnderlineSpan
,爲文本設置下劃線,具體實現方法以下:
SpannableString spannableString = new SpannableString("爲文字設置下劃線"); UnderlineSpan underlineSpan = new UnderlineSpan(); spannableString.setSpan(underlineSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE); textView.setText(spannableString);
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
,設置下標,功能與設置上標相似,不作過多描述,具體實現方法以下:
SpannableString spannableString = new SpannableString("爲文字設置下標"); SubscriptSpan subscriptSpan = new SubscriptSpan(); spannableString.setSpan(subscriptSpan, 5, spannableString.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE); textView.setText(spannableString);
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
,設置文本圖片,實現方法以下:
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
,設置可點擊的文本,設置這個屬性的文本能夠相應用戶點擊事件,至於點擊事件用戶能夠自定義,就像效果圖顯示同樣,用戶能夠實現點擊跳轉頁面的效果,具體實現方法以下:
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
,設置超連接文本,其實聰明的小夥幫在講到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拼接,很是實用。
看了這麼多的效果是否是收貨滿滿呢?最後我在附上一個個小小的彩蛋,小夥伴們能夠開動腦經想想是怎麼實現的哦! 要是有更好的創意,不妨在評論區留言跟大夥兒分享分享呢!