Android自定义View播放Gif动画的示例

前言

GIF是一种很常见的动态图片格式,在Android中它的使用场景非常多,大到启动页动画、小到一个Loading展示,都可以用GIF动画来完成,使用也很方便,直接从美工那边拿过来用就成。如果项目赶时间或者自定义原生动画太麻烦,GIF都是一个很好的选择,相比于最新的WEBP格式的动画,也有更好的兼容性(毕竟已经出现很多年了)。

关于图片加载我一直用的是Google推荐的 Glide ,图片加载和缓存都做的很好,同样也支持GIF动画。不过Glide默认就是循环播放Gif,没有开放相关的接口来控制Gif。这就使的我们不能很好地控制Gif的播放,比如控制播放开始时间、播放次数,播放暂停、播放开始、结束事件的监听,虽然用Glide可能做到(网上说可以,但我没找到方法),但操作也会很麻烦。

分析

除了第三方的库,Android自带的类 android.graphics.Movie 也可以用来加载播放Gif动画,而且实现起来很简单。

Movie decodeStream(InputStream is) Movie decodeFile(String pathName) Movie decodeByteArray(byte[] data, int offset,int length)

按来源分别可以从Gif文件的输入流,文件路径,字节数组中得到Movie的实列。然后我们可以通过操作Movie对象来操作Gif文件。

下面介绍下几个方法:

int width() movie的宽,值等于gif图片的宽,单位:px。

int height() movie的高,值等于gif图片的高,单位:px。

int duration() movie播放一次的时长,也就是gif播放一次的时长,单位:毫秒。

boolean isOpaque() Gif图片是否带透明

boolean setTime(int relativeMilliseconds) 设置movie当前处在什么时间,然后找到对应时间的图片帧,范围0 ~ duration。返回是否成功找到那一帧。

draw(Canvas canvas, float , float y) draw(Canvas canvas, float x, float y, Paint paint)

在Canves中画出当前帧对应的图像。x,y对应Movie左上角在Canves中的坐标。

以上就是Movie平常会用到大部分方法,下面就利用这些自定义VIew实现播放Gif动画。

实现

首先定义一些需要的属性,用于在布局文件中设置gif

<declare-styleable name="GIFVIEW"> <!--gif文件引用--> <attr name="gifSrc" format="reference" /> <!--是否加载完自动播放--> <attr name="authPlay" format="boolean" /> <!--播放次放,默认永远播放--> <attr name="playCount" format="integer" /> </declare-styleable>

然后定义Gifde的播放监听器,来监听各个时段的事件,都很简单就不再介绍了:

public interface OnPlayListener { void onPlayStart(); void onPlaying(int percent); void onPlayPause(boolean pauseSuccess); void onPlayRestart(); void onPlayEnd(); }

声明类,直接继承ImageView,这样我们不仅可以显示Gif动画,也可以显示普通图片:

public class GifImageView extends AppCompatImageView

然后加载Gif图片资源

public void setGifResource(int movieResourceId, OnPlayListener onPlayListener) { mOnPlayListener = onPlayListener; movie = Movie.decodeStream(getResources().openRawResource(movieResourceId)); if (movie == null) { //如果movie为空,那么就不是gif文件,尝试转换为bitmap显示 Bitmap bitmap = BitmapFactory.decodeResource(getResources(), movieResourceId); if (bitmap != null) { setImageBitmap(bitmap); return; } } movieDuration = movie.duration() == 0 ? DEFAULT_DURATION : movie.duration(); requestLayout(); }

调用requestLayout重新计算View大小,并重新绘制。

@Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { if (movie != null) { int movieWidth = movie.width(); int movieHeight = movie.height(); setMeasuredDimension(movieWidth, movieHeight); } else { super.onMeasure(widthMeasureSpec, heightMeasureSpec); } }

开始播放:

public void play(int counts) { this.counts = counts; reset(); if (mOnPlayListener != null) { mOnPlayListener.onPlayStart(); } invalidate(); }

不断调用onDraw方法来绘制Gif当前时间的图片帧:

@Override protected void onDraw(Canvas canvas) { if (movie != null) { if (!mPaused && hasStart) { drawMovieFrame(canvas); invalidateView(); } else { drawMovieFrame(canvas); } } else { super.onDraw(canvas); } } /** * 画出gif帧 */ private void drawMovieFrame(Canvas canvas) { movie.setTime(getCurrentFrameTime()); movie.draw(canvas, 0.0f, 0.0f); }

最核心的方法就是计算当前时间需要播放处于movie中的哪个时间段。

private int getCurrentFrameTime() { if (movieDuration == 0) return 0; //因为有暂停,所以需要减去暂停时间 long now = SystemClock.uptimeMillis() - dealyTime; int nowCount = (int) ((now - mMovieStart) / movieDuration); if (counts != -1 && nowCount >= counts) { hasStart = false; if (mOnPlayListener != null) { mOnPlayListener.onPlayEnd(); } } int currentTime = (int) ((now - mMovieStart) % movieDuration); int percent = currentTime * 100 / movieDuration; if (mOnPlayListener != null && hasStart) { mOnPlayListener.onPlaying(percent); } return currentTime; }

暂停Gif播放:

public void pause() { if (movie != null && !mPaused && hasStart) { mPaused = true; invalidate(); mMoviePauseTime = SystemClock.uptimeMillis(); if (mOnPlayListener != null) { mOnPlayListener.onPlayPause(true); } } else { if (mOnPlayListener != null) { mOnPlayListener.onPlayPause(false); } } }

继续Gif播放:

if (mPaused && mMoviePauseTime > 0) { mPaused = false; dealyTime = dealyTime + SystemClock.uptimeMillis() - mMoviePauseTime; invalidate(); if (mOnPlayListener != null) { mOnPlayListener.onPlayRestart(); } } 经过这些处理,我们就

能更好地控制Gif的播放流程了。下面简单看下成品图:

进阶

倒叙播放

相信看了上面GifImageView的实现原理后,倒叙播放的实现也是很容易的。

public void playReserver() { if (movie != null) { reset(); reverse = true; if (mOnPlayListener != null) { mOnPlayListener.onPlayStart(); } invalidate(); } } if (reverse) { movie.setTime(movieDuration - getCurrentFrameTime()); } else { movie.setTime(getCurrentFrameTime()); }

如下图,狗子的头已经从原来的左边转到右边变成了现在的右边转到左边(ಠᴗಠ)。

像播放视频一样播放Gif动画

这部分是我在写完GifView后想到的一点进阶功能,既然我们已经实现了播放和暂停,即能控制在某个时间点播放指定的Gif图片帧,如果再加入进度条,快进等功能,那么不就能做到和视频播放器一样的功能了吗?限于篇幅,我只简单实现了进度条功能,更多功能实现请移步Github,地址: GifView 。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

时间: 2024-08-02 14:29:31

Android自定义View播放Gif动画的示例的相关文章

Android 自定义view和属性动画实现充电进度条效果_Android

近期项目中需要使用到一种类似手机电池充电进度的动画效果,以前没学属性动画的时候,是用图片+定时器的方式来完成的,最近一直在学习动画这一块,再加上复习一下自定义view的相关知识点,所以打算用属性动画和自定义view的方式来完成这个功能,将它开源出来,供有需要的人了解一下相关的内容. 本次实现的功能类似下面的效果: 接下来便详细解析一下如何完成这个功能,了解其中的原理,这样就能举一反三,实现其他类似的动画效果了. 详细代码请看大屏幕 https://github.com/crazyandcoder

Android自定义View实现loading动画加载效果

项目开发中对Loading的处理是比较常见的,安卓系统提供的不太美观,引入第三发又太麻烦,这时候自己定义View来实现这个效果,并且进行封装抽取给项目提供统一的loading样式是最好的解决方式了. 先自定义一个View,继承自LinearLayout,在Layout中,添加布局控件 /** * Created by xiedong on 2017/3/7. */ public class Loading_view extends LinearLayout { private Context m

Android 自定义view和属性动画实现充电进度条效果

近期项目中需要使用到一种类似手机电池充电进度的动画效果,以前没学属性动画的时候,是用图片+定时器的方式来完成的,最近一直在学习动画这一块,再加上复习一下自定义view的相关知识点,所以打算用属性动画和自定义view的方式来完成这个功能,将它开源出来,供有需要的人了解一下相关的内容. 本次实现的功能类似下面的效果: 接下来便详细解析一下如何完成这个功能,了解其中的原理,这样就能举一反三,实现其他类似的动画效果了. 详细代码请看大屏幕 https://github.com/crazyandcoder

Android自定义View之绘制音乐播放器示波器

周末玩的有点嗨,没更新博客了,今天补上,这个示波器是在大学的时候老师教的,但是出来工作一直没有用到过,渐渐的也就忘记了,现在重新学习一下.来看看效果图: 这里是一个自定义的柱状图,然后有一个按钮,点击按钮的时候,这里柱子会不停的运动,类似于音乐播放器里示波器的跳动. 跟前面几个自定义view的方式类似,重写了onSizeChange()方法和onDraw()方法  先列一下我们要用到的变量 Paint mPaint; mWidth; mRectWidth; mRectHeight; mRectC

Android自定义view实现阻尼效果的加载动画_Android

效果: 需要知识: 1. 二次贝塞尔曲线 2. 动画知识 3. 基础自定义view知识 先来解释下什么叫阻尼运动 阻尼振动是指,由于振动系统受到摩擦和介质阻力或其他能耗而使振幅随时间逐渐衰减的振动,又称减幅振动.衰减振动.[1] 不论是弹簧振子还是单摆由于外界的摩擦和介质阻力总是存在,在振动过程中要不断克服外界阻力做功,消耗能量,振幅就会逐渐减小,经过一段时间,振动就会完全停下来.这种振幅随时间减小的振动称为阻尼振动.因为振幅与振动的能量有关,阻尼振动也就是能量不断减少的振动.阻尼振动是非简谐运

Android自定义View 实现闹钟唤起播放闹钟铃声功能_Android

先上图看一下闹钟唤期页面的效果 实现的功能: 1:转动的图片根据天气情况更换 2:转动时间可以设置,转动结束,闹铃声音就结束 3:光圈颜色渐变效果 直接上代码啦: package com.yuekong.sirius.extension.customview; import android.animation.Animator; import android.animation.ValueAnimator; import android.content.Context; import andro

Android自定义view实现阻尼效果的加载动画

效果: 需要知识: 1. 二次贝塞尔曲线 2. 动画知识 3. 基础自定义view知识 先来解释下什么叫阻尼运动 阻尼振动是指,由于振动系统受到摩擦和介质阻力或其他能耗而使振幅随时间逐渐衰减的振动,又称减幅振动.衰减振动.[1] 不论是弹簧振子还是单摆由于外界的摩擦和介质阻力总是存在,在振动过程中要不断克服外界阻力做功,消耗能量,振幅就会逐渐减小,经过一段时间,振动就会完全停下来.这种振幅随时间减小的振动称为阻尼振动.因为振幅与振动的能量有关,阻尼振动也就是能量不断减少的振动.阻尼振动是非简谐运

Android自定义View之仿vivo i管家病毒扫描动画效果

技术是永无止境的,如果真的爱技术,那就勇敢的坚持下去.我很喜欢这句话,当我在遇到问题的时候.当我觉得代码枯燥的时候,我就会问自己,到底是不是真的热爱技术,这个时候,我心里总是起着波澜,我的答案是肯定的,我深深的爱着这门技术. 今天我们继续聊聊Android的自定义View系列.先看看效果吧: 这个是我手机杀毒软件的一个动画效果,类似于雷达搜索,所以用途还是很广泛的,特别是先了解一下这里的具体逻辑和写法,对技术的进步一定很有用. 先简单的分析一下这里的元素,主要有四个圆.一个扇形.还有八条虚线.当

Android自定义View 画弧形,文字,并增加动画效果

一个简单的Android自定义View的demo,画弧形,文字,开启一个多线程更新ui界面,在子线程更新ui是不允许的,但是View提供了方法,让我们来了解下吧. 1.封装一个抽象的View类   BaseView.java /**    * 封装基本View   * @author ansen    * @create time 2015-08-07    */     public abstract class  BaseView extends View{         private