一种基于Glide图片加载框架的Android RichText实现

前言

在安卓中实现图文并茂的展示效果大体有两种方式:1.使用Android系统提供的WebView控件去直接展示一个HTML的网页 2.通过将HTML内容转化为Spanned格式在 TextView 中进行显示(也就是我们要讨论的一种)。虽然这两种方式都可以显示HTML内容,但是两者的实现过程,执行效率以及对用户交互的响应方式却有较大的不同。这些也决定了他们分别适合于不同的应用场景。一般来说,如果HTML的内容比较复杂,那还是建议使用WebView作为显示方式,因为TextView里面并不是支持所有的HTML标签,需要开发者额外增加对于标签的支持。这无异于增大了实现的复杂度。而对于显示格式化文本这样的需求,比如单纯图文混排这样的效果,使用TextView就再合适不过了,因为它相对于WebView更加轻量级,加载更高效。同时它也可以直接为图片和超链接提供点击事件。十分方便。好了,话不多说,先看看效果:

GIF_RichText.gif

过程

首先我们自定义控件RichText,让其继承TextView。在RichText中实现setRichText()方法:

public void setRichText(String text) {

        Spanned spanned = Html.fromHtml(text, new GlideImageGetter(getContext(), this), null);
        super.setText(spanned);

    }

该方法就是调用android.text.Html类提供的fromHtml()方法将传递进来的HTML内容转化为Spanned对象。Android.text.Html 类提供的 fromHtml()方法如下:

public static Spanned fromHtml (String source, Html.ImageGetter imageGetter, Html.TagHandler tagHandler)

source指的是传进来的HTML内容,它是由一些标签包裹内容组成的。一个简单的例子如下:

"<h1>RichText</h1><p>Android平台下的富文本解析器</p><img title=\"\"src=\"http://image.tianjimedia.com/uploadImages/2015/129/56/J63MI042Z4P8.jpg\"><br><br>"

第三个参数tagHandler是对HTML内容中特殊标签的支持。我们此处可以设为null。
我们可以看出文本和超链接内容,TextView是可以通过读标签及其内容来直接显示。那么图片怎么办呢?而第二个参数Html.ImageGetter就是用来获取图片资源的。
ImageGetter只是一个接口,我们需要实现它的如下方法:

public Drawable getDrawable(String url) {}

该方法的参数url即为图片资源的URL值,是Html类将读取到的<img>标签下src值传递过来的。而我们这里要做的只是将对应URL的图片资源Drawable返回即可(是不是很简单)。读取网络图片资源,我们当然优先使用Glide了,为什么呢?因为它确实很好用啊,并且还支持GIF哦。不了解的小朋友可以看以下两个传送门:

关于Glide的基本使用请参考这篇使用Glide加载图片系列之一从不同的数据源加载图片
关于Glide与Picasso的对比请参考这篇Google推荐的图片加载库Glide介绍
另外Android大神stormzhang也觉得它很好的Android开源项目推荐之「图片加载到底哪家强」

接下来就是实现getDrawable方法了:

@Override
    public Drawable getDrawable(String url) {
        final UrlDrawable urlDrawable = new UrlDrawable();
        final GenericRequestBuilder load;
        final Target target;
        if(isGif(url)){
            load = Glide.with(mContext).load(url).asGif();
            target = new GifTarget(urlDrawable);
        }else {
            load = Glide.with(mContext).load(url).asBitmap();
            target = new BitmapTarget(urlDrawable);
        }
        targets.add(target);
        load.into(target);
        return urlDrawable;
    } 

上面的方法主要完成以下几点:
1.生成要返回的urlDrawable对象。
2.判断url是否指向一个GIF图片资源。(其实就是判断这个字串的结尾是否包含.gif而已)
3.根据图片资源的不同,通过Glide产生不同的GenericRequestBuilder(可以理解为包含不同资源Drawable的数据源)
4.生成不同的数据载体。用来接收GIFDrawable或者Bitmap.
5.将数据源注入载体。并通过载体的回调方法为urlDrawable赋值。
6.收集target以便在合适的机会下释放掉内存。
6.最后返回urlDrawable。

UrlDrawable的实现比较简单,如下:

class UrlDrawable extends BitmapDrawable{
    private Drawable drawable;

    @SuppressWarnings("deprecation")
    public UrlDrawable() {
    }
    @Override
    public void draw(Canvas canvas) {
        if (drawable != null)
            drawable.draw(canvas);
    }
    public Drawable getDrawable() {
        return drawable;
    }
    public void setDrawable(Drawable drawable) {
        this.drawable = drawable;
    }
}

GifTarget.java的实现如下所示:

private class GifTarget extends SimpleTarget<GifDrawable> {
       private final UrlDrawable urlDrawable;
       private  GifTarget(UrlDrawable urlDrawable) {
           this.urlDrawable = urlDrawable;
       }
       @Override
       public void onResourceReady(GifDrawable resource, GlideAnimation<? super GifDrawable> glideAnimation) {
           int w = MeasureUtil.getScreenSize(mContext).x;
           int hh=resource.getIntrinsicHeight();
           int ww=resource.getIntrinsicWidth() ;
           int high = hh * (w - 50)/ww;
           Rect rect = new Rect(20, 20,w-30,high);
           resource.setBounds(rect);
           urlDrawable.setBounds(rect);
           urlDrawable.setDrawable(resource);
           gifDrawables.add(resource);
           resource.setCallback(mTextView);
           resource.start();
           resource.setLoopCount(GlideDrawable.LOOP_FOREVER);
           mTextView.setText(mTextView.getText());
           mTextView.invalidate();
       }
   } 

而 BitmapTarget的具体实现如下:

private class BitmapTarget extends SimpleTarget<Bitmap> {
       private final UrlDrawable urlDrawable;
       public BitmapTarget(UrlDrawable urlDrawable) {
           this.urlDrawable = urlDrawable;
       }
       @Override
       public void onResourceReady(Bitmap resource, GlideAnimation<? super Bitmap> glideAnimation) {
           Drawable drawable = new BitmapDrawable(mContext.getResources(), resource);
           int w = MeasureUtil.getScreenSize(mContext).x;
           int hh=drawable.getIntrinsicHeight();
           int ww=drawable.getIntrinsicWidth() ;
           int high=hh*(w-50)/ww;
           Rect rect = new Rect(20, 20,w-30,high);
           drawable.setBounds(rect);
           urlDrawable.setBounds(rect);
           urlDrawable.setDrawable(drawable);
           mTextView.setText(mTextView.getText());
           mTextView.invalidate();
       }
   } 

可以发现以上两个Target的实现比较类似,只是可以提供的数据类型的不同,以及对拿到的resource处理的方式不一样。程序的入口是onResourceReady方法,我们拿到resource后,首先设置它要显示位置的边界。我们这里默认将每一个图片的宽度设置为屏幕宽度左右各减去20.高度按照原始宽高进行换算。然后为urlDrawable赋值,并刷新TextView。这样静态图片就能显示了。但是GIF还不行。因为GIF需要连续的View重绘才行。所以我们需要为GIF的drawable设置Drawable.CallBack回调。然后在回调函数里单独对TextView进行刷新动作。具体如下:

@Override
public void invalidateDrawable(Drawable who) {    
 Log.e("Text", "text is refreash");
 mTextView.invalidateOutline();
}

PS:上面的刷新代码我们最好不要使用mTextView.invalidate()这个方法,因为它会导致UI滑动时的卡顿。

接下来我们还可以在setRichText方法中为图片提供点击响应(超链接TextView默认是支持的)完成的代码如下:

 public void setRichText(String text) {
        Spanned spanned = Html.fromHtml(text, new GlideImageGetter(getContext(), this), null);
        SpannableStringBuilder spannableStringBuilder;
        if (spanned instanceof SpannableStringBuilder) {
            spannableStringBuilder = (SpannableStringBuilder) spanned;
        } else {
            spannableStringBuilder = new SpannableStringBuilder(spanned);
        }
        // 处理图片得点击事件
        ImageSpan[] imageSpans = spannableStringBuilder.getSpans(0, spannableStringBuilder.length(), ImageSpan.class);
        final List<String> imageUrls = new ArrayList<>();
        for (int i = 0, size = imageSpans.length; i < size; i++) {
            ImageSpan imageSpan = imageSpans[i];
            String imageUrl = imageSpan.getSource();
            int start = spannableStringBuilder.getSpanStart(imageSpan);
            int end = spannableStringBuilder.getSpanEnd(imageSpan);
            imageUrls.add(imageUrl);
            final int finalI = i;
            ClickableSpan clickableSpan = new ClickableSpan() {
                @Override
                public void onClick(View widget) {
                    if (onRichTextImageClickListener != null) {
                        onRichTextImageClickListener.imageClicked(imageUrls, finalI);
                    }
                }
            };
            ClickableSpan[] clickableSpans = spannableStringBuilder.getSpans(start, end, ClickableSpan.class);
            if (clickableSpans != null && clickableSpans.length != 0) {
                for (ClickableSpan cs : clickableSpans) {
                    spannableStringBuilder.removeSpan(cs);
                }
            }
            spannableStringBuilder.setSpan(clickableSpan, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        }
        super.setText(spanned);
        setMovementMethod(LinkMovementMethod.getInstance());
    }
    public void setOnRichTextImageClickListener(OnRichTextImageClickListener onRichTextImageClickListener) {
        this.onRichTextImageClickListener = onRichTextImageClickListener;
    }
    public interface OnRichTextImageClickListener {
        /**
         * 图片被点击后的回调方法
         *
         * @param imageUrls 本篇富文本内容里的全部图片
         * @param position  点击处图片在imageUrls中的位置
         */
        void imageClicked(List<String> imageUrls, int position);
    }

为RichText绑定图片点击操作:

 mBodyTv.setOnRichTextImageClickListener(new RichText.OnRichTextImageClickListener(){
    @Override
    public void imageClicked(List<String> imageUrls, int position){
        Toast.makeText(MainActivity.this, imageUrls.get(position),Toast.LENGTH_SHORT).show();
    }

});

最后附上这个Demo的源码吧——github地址

希望在Android学习的路上,大家共同成长!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,980评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,422评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,130评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,553评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,408评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,326评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,720评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,373评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,678评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,722评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,486评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,335评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,738评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,283评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,692评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,893评论 2 335

推荐阅读更多精彩内容