原生Android控件ImageView并不能支持播放GiF格式的图片.如果将一张GIF的图片放入ImageView中,它只会显示图片的第一帧,不会产生任何动画效果.

Android中播放GIF动画实现方法还是用多种的,最常用的就是使用   Frame动画, 但局限性较多,所以下面用一种拓展的ImageView实现效果.

    
1.要用到自定义控件,就要使用自定义控件的属性,因此需要在values下新建一个attrs.xml,可以为这个文件中添加任何需要自定义的属性.
这里只需要一个auto_play,标志是否自动播放.
<?xml version="1.0" encoding="utf-8"?>  
<resources>  
  
    <declare-styleable name="PowerImageView">  
        <attr name="auto_play" format="boolean"></attr>  
    </declare-styleable>  
  

</resources>

2.下面开始写最重要的MyImageView继承自ImageView.
public class MyImageView extends ImageView implements OnClickListener {
    
    private Movie mMovie;   //播放GIF动画的关键类
    
    private Bitmap mStartBitmap;   //播放按钮
    
    private long mMovieStart;  //动画开始的时间
    private int mImageWidth;  //GIF图片的宽度
    
    private int mImageHeight;  //GIF图片的高度
    
    private boolean isPlaying;   //标志是否正在播放
    
    private boolean isAutoPlay;  //是否自动播放
    
    
    public MyImageView(Context context, AttributeSet attrs) {
        super(context, attrs);
        TypedArray a=context.obtainStyledAttributes(attrs, R.styleable.PowerImageView);
        int resourceId=getResourceId(a, context, attrs);
        if(resourceId!=0){
            //获取该资源的流
            InputStream is=getResources().openRawResource(resourceId);
            //使用Movie对流进行解码
            mMovie=Movie.decodeStream(is);
            if(mMovie!=null){
                //如果返回不为空,则说明这是一个GIF图片,下面获取自动播放的属性
                isAutoPlay=a.getBoolean(R.styleable.PowerImageView_auto_play,false);
                Bitmap bitmap=BitmapFactory.decodeStream(is);
                mImageWidth=bitmap.getWidth();
                mImageHeight=bitmap.getHeight();
                bitmap.recycle();
                
                if(!isAutoPlay){
                    mStartBitmap=BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher);
                    setOnClickListener(this);
                }
            }
        }
        
    
    }
    
    
    
    //通过Java反射,获取到src指定图片资源所对应的id。
    private int getResourceId(TypedArray a, Context context,AttributeSet attrs){
        try {
            Field field =TypedArray.class.getDeclaredField("mValue");
            field.setAccessible(true);
            TypedValue typeValueObject=(TypedValue) field.get(a);
            return typeValueObject.resourceId;
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        }
            return 0;
    }
    
    
    @Override
    public void onClick(View v) {
        if(v.getId()==getId()){
            isPlaying=true;
            invalidate();
        }
    }
    
    @Override
    protected void onDraw(Canvas canvas) {
        if(mMovie==null){
            super.onDraw(canvas);
        }
        else{
            if(isAutoPlay){
                playMovie(canvas);
                invalidate();
            }
            else{
                if(isPlaying){
                    if(playMovie(canvas)){
                        isPlaying=false;
                    }
                    invalidate();
                }
                else{
                    mMovie.setTime(0);
                    mMovie.draw(canvas, 0, 0);
                    int offsetW=(mImageWidth-mStartBitmap.getWidth())/2;
                    int offsetH=(mImageHeight=mStartBitmap.getHeight())/2;
                    canvas.drawBitmap(mStartBitmap, offsetW, offsetH,null);
                }
            }
        }
        
    }
    
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        if(mMovie!=null){
            setMeasuredDimension(mImageWidth, mImageHeight);
        }
    }
    
    //开始播放GIF动画   当播放结束返回true.
    private boolean playMovie(Canvas canvas){
        long now=SystemClock.uptimeMillis();   //获取从手机开机到现在的毫秒数
        if(mMovieStart==0){
            mMovieStart=now;
        }
        int duration=mMovie.duration();
        if(duration==0){
            duration=1000;
        }
        int relTime=(int) ((now - mMovieStart) % duration);
        mMovie.setTime(relTime);
        mMovie.draw(canvas, 0, 0);
        if((now-mMovieStart)>=duration){
            mMovieStart=0;
            return true;
        }
        return false;
    }
    

}  

上面代码中,playMovie是播放动画的关键方法.
3.activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:attr="http://schemas.android.com/apk/res/com.example.powerimageviewdemo" 
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >
    <com.example.powerimageviewdemo.MyImageView 
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:src="@drawable/vv"
        attr:auto_play="false"
        />

</RelativeLayout>

使用src属性可以指定任意一张图片,如果是GIF图片,则会进行播放,如果auto_play设置的是false,则不会自动播放.
有些手机启用了硬件加速功能后导致GIF动画播放不了.需要在AndroidManifest.xml 进行配置.

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.mydemo"
  4. android:versionCode="1"
  5. android:versionName="1.0" >
  6. <uses-sdk
  7. android:minSdkVersion="14"
  8. android:targetSdkVersion="17" />
  9. <application
  10. android:allowBackup="true"
  11. android:icon="@drawable/ic_launcher"
  12. android:label="@string/app_name"
  13. android:theme="@style/AppTheme"
  14. android:hardwareAccelerated="false"
  15. >
  16. <activity
  17. android:name="com.example.mydemo.MainActivity"
  18. android:label="@string/app_name" >
  19. <intent-filter>
  20. <action android:name="android.intent.action.MAIN" />
  21. <category android:name="android.intent.category.LAUNCHER" />
  22. </intent-filter>
  23. </activity>
  24. </application>
  25. </manifest>

 

 

android 拓展ImageView播放GIF动画的更多相关文章

  1. Android(java)学习笔记198:Android下的逐帧动画(Drawable Animation)

    1.帧动画: 帧动画顾名思义,一帧一帧播放的动画就是帧动画. 帧动画和我们小时候看的动画片的原理是一样的,在相同区域快速切换图片给人们呈现一种视觉的假象感觉像是在播放动画,其实不过是N张图片在一帧一帧 ...

  2. Android(java)学习笔记141:Android下的逐帧动画(Drawable Animation)

    1. 帧动画: 帧动画顾名思义,一帧一帧播放的动画就是帧动画. 帧动画和我们小时候看的动画片的原理是一样的,在相同区域快速切换图片给人们呈现一种视觉的假象感觉像是在播放动画,其实不过是N张图片在一帧一 ...

  3. Android实现播放GIF动画的强大ImageView

    我个人是比较喜欢逛贴吧的,贴吧里总是会有很多搞笑的动态图片,经常看一看就会感觉欢乐很多,可以释放掉不少平时的压力.确实,比起一张单调的图片,动态图片明显更加的有意思.一般动态图片都是GIF格式的,浏览 ...

  4. Android播放gif动画,增加屏幕掉金币效果

    前言:播放gif的版本有很多,我这边使用Android自带的Movie类播放gif动画,也是在别人的基础上进行修改.有同样需求的朋友可以参考我的demo. 1.效果图如下: 2.部分主要代码 Main ...

  5. android 通过帧动画方式播放Gif动画

    注意:经过本人测试,这个方法很耗内存, 图片一多就崩了.慎用 <1>用工具(photoshop或者FireWorks)将GIF动画图片分解成多个GIF静态图片,然后保存在res\drawa ...

  6. Android 播放Gif 动画

    在Android 中是不支持直接使用Gif 图片关联播放帧动画,如下动画在Android 中是无法播放的: Android 提供了另外一种解决的办法,就是使用AnimationDrawable 这一函 ...

  7. Android只播放gif动画

    使用easygifanimator软件把gif动画打散为图片. 第一步:先上图片素材,以下素材放到res/drawable目录下: 转:http://blog.csdn.net/aminfo/arti ...

  8. Android高级控件(二)——SurfaceView实现GIF动画架包,播放GIF动画,自己实现功能的初体现

    Android高级控件(二)--SurfaceView实现GIF动画架包,播放GIF动画,自己实现功能的初体现 写这个的原因呢,也是因为项目中用到了gif动画,虽然网上有很多的架包可以实现,不过我们还 ...

  9. 动画_ _ Android应用开发之所有动画使用详解

    转载: http://blog.csdn.net/yanbober/article/details/46481171 题外话:有段时间没有更新博客了,这篇文章也是之前写了一半一直放在草稿箱,今天抽空把 ...

随机推荐

  1. 如何安装Ruby(Windows)

    Ruby解释器的安装 1.Windows平台 想尽快安装并运行Ruby,可遵循如下步骤: 1.启动Web浏览器,访问 http://www.ruby-lang.org/en/downloads/ 2. ...

  2. caffe新版本的各种软件

    系统重装了,于是,我想装就体验一下最新的各种东西吧. anaconda最新的 cuda最新的 cudnn最新的 本来安装好了没问题.caffe编译也通过了.但是不能用,缺少python opencv和 ...

  3. js世界这么大,闭包想看看

    什么是闭包,为什么要用他?闭包是能够访问其他函数作用域的函数.我们来分析下句子成分(语文大神),闭包是函数,js函数的作用域分为全局作用域,局部作用域,eval作用域,并没有块级作用域形象的讲,每个函 ...

  4. cookie和session的介绍

    1.cookie和session cookie不属于http协议范围,由于http协议无法保持状态,但实际情况,我们却又需要“保持状态”,因此产生cookie. cookie的工作原理是:由服务器产生 ...

  5. JQuery制作网页—— 第三章 JavaScript操作DOM对象

    1. DOM:Document Object Model(文档对象模型):          DOM操作:                   ●DOM是Document Object Model的缩 ...

  6. 最小化 Java 镜像的常用技巧

    背景 随着容器技术的普及,越来越多的应用被容器化.人们使用容器的频率越来越高,但常常忽略一个基本但又非常重要的问题 - 容器镜像的体积.本文将介绍精简容器镜像的必要性并以基于 spring boot ...

  7. php学习【1】

    1:输出语句 <?php echo "hellow world"; print "hellow world"; print_r ("helow ...

  8. SQLSERVER 数据库恢复挂起的解决办法

    如果你的数据库还处于挂起状态,请把我下面代码的test改为你的库名,然后执行完,刷新就正常了: USE masterGOALTER DATABASE test SET SINGLE_USERGOALT ...

  9. 【Effective C++ 读书笔记】条款03: 尽量使用 const

    关键字const多才多艺,变化多端却不高深莫测. const 修饰指针 面对指针, 你可以指出 指针自身.指针所指物.或者两者都不是 const. 如果关键字 const 出现在星号左边,表示被指物是 ...

  10. &、|、~与&&、||、! 谬误

    按位运算符(&.|.~)的操作是被默认为一个二进制的位序列,分别对其中的每个位进行操作. 逻辑运算符(&&.||.!)将操作数当成非真及假,非假及真.通常就是将0当成假,非0即 ...