TextView是不只可以String的,我们平常用的给setText()方法传递String参数的时候,其实是调用的public final void setText (CharSequence text)
方法,String类是CharSequence的子类。
而CharSequence子类众多,其中有一个接口Spanned,即类似html的带标记的文本。我们可以用它来在TextView中显示html(自然,有很多html标记是不支持的,只支持一部分)。
android.text.Html类的一个方法:public static Spanned fromHtml (String source)
可以将html代码转换为Spanned。
1 2 3 4 5 html = "<h1>this is h1</h1>" + "<p>This text is normal</p>" + "<img src='https://www.google.com.hk/intl/zh-CN/images/logo_cn.png' />" ; Spanned sp = Html . fromHtml(html ) ; textView.setText(sp ) ;
显示效果: 可以看出,字体效果基本是显示出来了,但是图片没有显示。
要实现图片的显示需要使用Html.fromHtml的另外一个重构方法:
public static Spanned fromHtml (String source, Html.ImageGetter imageGetter, Html.TagHandler tagHandler)
其中Html.ImageGetter
是一个接口,我们要实现此接口,在它的getDrawable(String source)
方法中返回图片的Drawable对象才可以。
修改后的代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 Spanned sp = Html . fromHtml(html , new Html.ImageGetter() { @Override public Drawable getDrawable(String source ) { InputStream is = null; try { is = (InputStream) new URL(source ) .getContent() ; Drawable d = Drawable . createFromStream(is , "src" ) ; d.setBounds(0, 0, d .getIntrinsicWidth () , d.getIntrinsicHeight() ); is.close() ; return d; } catch (Exception e) { return null; } } }, null); textView.setText(sp ) ;
看起来有些复杂,但其实只是fromHtml()
的第二个参数是一个匿名类,用以图片的获取。
其中
1 2 is = (InputStream) new URL(source ) .getContent() ; Drawable d = Drawable . createFromStream(is , "src" ) ;
用以通过图片的地址获取相应的Drawable实例。
由于用到了网络资源的图片,所以要在Mainifest文件中加入权限:<uses-permission android:name="android.permission.INTERNET" />
修改后的运行结果:
图片正常显示了。
注意:通过网络获取图片是一个耗时的操作,最好不要放在主线程中,否则容易引起阻塞。在高版本的Android系统中,显示图片失效,因为在主线程中无法加载网络图片,此时可以考虑使用Glide加载图片。
通过上面的代码我们知道要想展示图片,我们可以重写fromHtml()
的第二个参数ImageGetter:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 public class MImageGetter implements ImageGetter{ Context c; TextView container; public MImageGetter(TextView text ,Context c ) { this.c = c; this.container = text; } public Drawable getDrawable(String source ) { final LevelListDrawable drawable = new LevelListDrawable() ; Glide .with (c).load(source).as Bitmap() .into(new SimpleTarget<Bitmap>() { @Override public void onResourceReady(Bitmap resource , GlideAnimation<? super Bitmap> glideAnimation ) { if (resource != null) { BitmapDrawable bitmapDrawable = new BitmapDrawable(resource ) ; drawable.addLevel(1, 1, bitmapDrawable ) ; drawable.setBounds(0, 0, resource .getWidth () ,resource.getHeight() ); drawable.setLevel(1) ; container.invalidate() ; container.setText(container .getText () ); } } }); return drawable; } }
加载图片的时候我用的是glide异步加载,其他原理一样。现在图片已经可以正常展示出来了。
上面介绍的是显示网络上的图片,但如何显示本地的图片呢:
1 2 3 4 5 6 7 8 ImageGetter imgGetter = new Html.ImageGetter() { public Drawable getDrawable(String source ) { Drawable drawable = null; drawable = Drawable . createFromPath(source ) ; drawable.setBounds(0, 0, drawable .getIntrinsicWidth () , drawable.getIntrinsicHeight() ); return drawable; } };
只需将source改为本地图片的路径便可,在这里我使用的是:
1 2 String source ; source =getFilesDir()+"/ic_launcher.png" ;
参考:Android TextView使用HTML处理字体样式、显示图片等 TextView显示html图片的方法 Android textView展示html图片,实现图文混排,点击查看大图片