原文转载自http://chentingjia.iteye.com/blog/822710
昨晚研读 ApiDemo 源码至 com.example.android.apis.text.Link 类。首先,看一下其运行效果:
要给 TextView 加上效果,方式主要有几种:
第一种,自动应用效果,使用 android:autolink 属性,如:
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/text1"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:autoLink="all"
android:text="@string/link_text_auto"
/>
第二种,在文本中使用 <a> 标签,如:
<string name="link_text_manual"><b>text2:</b> This is some other
text, with a <a href="http://www.google.com">link</a> specified
via an <a> tag. Use a \"tel:\" URL
to <a href="tel:4155551212">dial a phone number</a>
</string>
第三种,和第二种其实是一样的,只不过将文本改在 JAVA 代码中,如:
TextView t3 = (TextView) findViewById(R.id.text3);
t3.setText(
Html.fromHtml(
"<b>text3:</b> Text with a " +
"<a href=\"http://www.google.com\">link</a> " +
"created in the Java source code using HTML."));
t3.setMovementMethod(LinkMovementMethod.getInstance());
第四种,前面三种可以说都是自动的,而第四种就是纯“手工”的了。通过创建 SpanableString 字符串,并在之上创 建一个或多个 Span 来实现丰富的效果。例子如下:
SpannableString ss = new SpannableString("text4: Click here to dial the phone.");
ss.setSpan(new StyleSpan(Typeface.BOLD), 0, 6,Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
ss.setSpan(new URLSpan("tel:4155551212"), 13, 17,Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
TextView t4 = (TextView) findViewById(R.id.text4);
t4.setText(ss);
t4.setMovementMethod(LinkMovementMethod.getInstance());
完整的代码见 ApiDemo 吧,下面我提几点需要注意的:
- setMovementMethod,此方法在需要响应用户事件时使用,如点击一个电话号码就跳转到拨号页面。如果不执行这个方法是不会响应事件的,即便文本看着已经是下划线蓝色字了。
- Spanned.SPAN_EXCLUSIVE_EXCLUSIVE,这是在 setSpan 时需要指定的 flag,它的意义我试了很久也没试出来,睡个觉,今天早上才突然有点想法,试之,果然。它是用来标识在 Span 范围内的文本前后输入新的字符时是否把它们也应用这个效果。分别有 Spanned.SPAN_EXCLUSIVE_EXCLUSIVE(前后都不包括)、Spanned.SPAN_INCLUSIVE_EXCLUSIVE(前面包括,后面不包括)、Spanned.SPAN_EXCLUSIVE_INCLUSIVE(前面不包括,后面包括)、Spanned.SPAN_INCLUSIVE_INCLUSIVE(前后都包括)。看个截图就更明白了:由于TextView中的文字内容没有"插入"一说,所以这Spanned.*的flag用在TextView上是没有什么效果的;在EditText中的效果如下图,其中"Click"是Spanned.SPAN_EXCLUSIVE_INCLUSIVE的,"here"是Spanned.SPAN_EXCLUSIVE_EXCLUSIVE的,"dial"是Spanned.SPAN_INCLUSIVE_INCLUSIVE的

对比一下
-------------------------------------------------------------------------------------------------------------------------------------------
上面第四种方法用到的Span有多种现成的可用,所有Span都继承自抽象类android.text.style.CharacterStyle,常见的Span有
- ForegroundColorSpan:用于控制文字颜色
- URLSpan:用于响应Uri,如"tel:911"
- StyleSpan:用于控制文字风格(粗体、斜体、粗斜体、正常),文字可用风格定义在android.graphics.Typeface类中
- UnderlineSpan:用于文字下划线
- StrikethroughSpan:用于文字中划线(也叫删除线)
SpannableString.setSpan(Object what, int start, int end, int flags);