美文网首页Android组件开发技巧文章
一种基于Glide图片加载框架的Android RichText

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

作者: 睡在向日葵 | 来源:发表于2016-06-24 20:52 被阅读2606次

前言

在安卓中实现图文并茂的展示效果大体有两种方式: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学习的路上,大家共同成长!

相关文章

网友评论

  • andcoder:当图片很多时,是会有OOM产生的,所以这种封装还是有问题
  • 71a421f4aa69:为什么我的图片加载不出来?我写成了静态方法。
  • 56b0a525a3b4:非wifi状态下不加载图片,然后点击重新加载,应该这么去改动
  • 炫丽oo人生:请问加载多张gif图 gif 乱了 怎么办 几张图的的画面 每个gif上都有,希望快点回复,急 谢谢
  • xzmoji:非常有帮助,感谢!
  • 帅气的猪猪:如何用tagHandler读取 “ data-original ”标签里面的图片链接呢?
    <p>小罗37岁生日,感谢他陪我度过…最初看球的一段时光。<img class="sc-img img-jpg lazy" style-data="width:7.1rem;height:3.9250943396226416rem" src="http://7xpd1v.com2.z0.glb.qiniucdn.com/1_1473758890_1473758923993.png?imageView2/3/w/710&quot; data-original="http://ucdn.qiudd.net/FpP3j112z1bcfGrQ_AyWu3E0nLrC&quot;></p><p>
  • 367053bacd51:楼主,如果html中含有多张图片,除了第一张和最后一张是正常的,其他都会有点击不对应的问题。position=1的图片总是显示pos=0的,pos=2的显示pos1的依次到倒数第二张
    367053bacd51:@睡在向日葵 还有一个问题,这样的没法显示长图,会占空间但是不显示
    367053bacd51:@睡在向日葵 格式是<a href="原图链接"><img src="缩略图链接"></a>,我在</a>之后增加了<br />就可以了,具体原因不知道,暂时只要<p><a href><img></a></p>或者br都能正常
    睡在向日葵:@用户6515147083 是否是你html格式的问题导致系统解析图片资源生成ImageSpan时发生了错位的现象呢?
  • wphper:请问这个GlideImageGetter类是怎么来的?我项目中引用了Glide 3.7
    睡在向日葵: @wphper 这个类是自己实现的。不是glide库中包含的。
  • 30d510f59682:有解决自动换行问题不?
    睡在向日葵:@cbbs 自动换行跟你传进去的 HTML文本内容有关的,只要有换行符就行了。
  • Backaway:牛逼啊:smile:

本文标题:一种基于Glide图片加载框架的Android RichText

本文链接:https://www.haomeiwen.com/subject/qzindttx.html