【Android实战】记录自学自定义GifView过程,能同时支持gif和其他图片!【实用篇】

本文记录了自定义Android GifView以支持播放GIF和其他图片格式的过程。通过反射和Movie类解决GIF支持问题,使用android-gif-drawable库提高效率。此外,还分享了解决Movie.decodeFile不工作和动态设置图片的解决方案。

之前写了一篇博客,《【Android实战】记录自学自定义GifView过程,详解属性那些事!【学习篇】》
关于自定义GifView的,详细讲解了学习过程及遇到的一些类的解释,然后完成了一个项目,能通过在xml加入自定义 view (MyGifView)中加入自定义属性(my:gif_src = “@drawable/coffee”),达到播放gif图片的效果。


但是,有几个问题

1.gif_src 属性只支持 gif 图,并不支持其他类型的图片

2.只支持默认的引用图片,不能另外设置


问题一

gif_src 属性只支持 gif 图,并不支持其他类型的图片。

解决思路:
ImageView本身有个属性 src 是定义好的,已经可以用它播放静态图片,如果再能通过它播放动态图片,不就解决问题啦?!

于是查看 ImageView 类的源码,看到构造函数

public ImageView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
    //...

    final TypedArray a = context.obtainStyledAttributes(
                    attrs, com.android.internal.R.styleable.ImageView, defStyleAttr, defStyleRes);

    Drawable d = a.getDrawable(com.android.internal.R.styleable.ImageView_src);
    if (d != null) {
         setImageDrawable(d);
    }
    //...
}

有没有很眼熟?!对,之前自定义属性的时候用过!这里不过把属性路径改了!之前我们用的是自定义的路径 R.styleable.GifView 。

于是乎,我也想着,要是能在继承类(MyGifView)里面复制上段代码,然后再用movie转化,转换成功说明是 gif,就用之前的方法播放,转换失败说明是其他格式的图片,就交给 ImageView 自己处理!

真是好办法!

然而,根本不能这么用:

这里写图片描述

不能导入internal的包!

但是思路是对的!

通过参考《 Android PowerImageView实现,可以播放动画的强大ImageView》

得知了可以用反射!
中心代码:

  /** 
     * 通过Java反射,获取到src指定图片资源所对应的id。 
     *  
     * @param a 属性组
     * @param context 
     * @return 返回布局文件中指定图片资源所对应的id,没有指定任何图片资源就返回0。 
     */  
    private int getResourceId(TypedArray a, Context context) {  
        try {  
            Field field = TypedArray.class.getDeclaredField("mValue");  
            field.setAccessible(true);  
            TypedValue typedValueObject = (TypedValue) field.get(a);  
            return typedValueObject.resourceId;  
        } catch (Exception e) {  
            e.printStackTrace();  
        } finally {  
            if (a != null) {  
                a.recycle();  
            }  
        }  
        return 0;  
    }  

之前在自定义view初始化中的代码,我是用得到自定义属性值的方法获取gif的数据

int resId = typedArray.getResourceId(R.styleable.GifView_gif_src, 0); //gif_src属性对应值

现在只需要改这一句就好啦!

//int resId = typedArray.getResourceId(R.styleable.GifView_gif_src, 0); //gif_src属性对应值

int resId = getResourceId(typedArray, context); //src属性对应值

然后后面都不用改啦!

(但是转换成 InputStream 的时候,还是要加一句判断 if (resId != 0)再进行转换)

if (resId != 0) {
    InputStream iStream = getResources().openRawResource(resId); //此方法能通过资源文件id查找到资源文件并转化为输入流
    mMovie = Movie.decodeStream(iStream); //输入流转化为Movie (mMovie 为全局变量,类型 Movie)
}

好了问题解决!现在可以在xml用src属性指向.gif文件,并且进行正常播放了!

问题二

只支持默认的引用图片,不能另外设置

解决思路:
从外面设置无非就是外面调用setImageResource(int resId)setImageDrawable(Drawable drawable)setImageBitmap(Bitmap bm)等这些方法去改变 ImageView 属性 src 所对应的值!

那么,重写这些方法,把资源改成我们的 movie 就好啦!so easy!

首先重写setImageResource(int resId)

@Override 
public void setImageResource(int resId) {
    if (resId != 0) {
        InputStream iStream = getResources().openRawResource(resId);
        setMovie(iStream);
        if (mMovie == null) {
            super.setImageResource(resId);
        }
    } else {
        super.setImageResource(re
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值