gif图动画在android中还是比较常用的,比如像新浪微博中,有很多gif图片,而且展示非常好,所以我也想弄一个。经过我多方的搜索资料和整理,终于弄出来了,其实github上有很多开源的gif的展示代码,我下载过几个,但是都不是很理想,不是我完全想要的。所以有时候就得自己学会总结,把开源的东西整理成自己的,现在无聊,也正好有朋友需要,所以现在整理了一下,留着以后备用!

废话不多说,直接上图:

在这里主要用的是:android中的android.graphics.Movie 这个类,这是android提供给我们的一个非常方便的工具。

首先,重写控件View,自定义一个展示gif图的GifView,代码如下:

[java] view
plain
copy

  1. package net.loonggg.gif.view;
  2. import net.loonggg.gif.R;
  3. import android.annotation.SuppressLint;
  4. import android.content.Context;
  5. import android.content.res.TypedArray;
  6. import android.graphics.Canvas;
  7. import android.graphics.Movie;
  8. import android.os.Build;
  9. import android.util.AttributeSet;
  10. import android.view.View;
  11. public class GifView extends View {
  12. /**
  13. * 默认为1秒
  14. */
  15. private static final int DEFAULT_MOVIE_DURATION = 1000;
  16. private int mMovieResourceId;
  17. private Movie mMovie;
  18. private long mMovieStart;
  19. private int mCurrentAnimationTime = 0;
  20. private float mLeft;
  21. private float mTop;
  22. private float mScale;
  23. private int mMeasuredMovieWidth;
  24. private int mMeasuredMovieHeight;
  25. private boolean mVisible = true;
  26. private volatile boolean mPaused = false;
  27. public GifView(Context context) {
  28. this(context, null);
  29. }
  30. public GifView(Context context, AttributeSet attrs) {
  31. this(context, attrs, R.styleable.CustomTheme_gifViewStyle);
  32. }
  33. public GifView(Context context, AttributeSet attrs, int defStyle) {
  34. super(context, attrs, defStyle);
  35. setViewAttributes(context, attrs, defStyle);
  36. }
  37. @SuppressLint("NewApi")
  38. private void setViewAttributes(Context context, AttributeSet attrs,
  39. int defStyle) {
  40. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
  41. setLayerType(View.LAYER_TYPE_SOFTWARE, null);
  42. }
  43. // 从描述文件中读出gif的值,创建出Movie实例
  44. final TypedArray array = context.obtainStyledAttributes(attrs,
  45. R.styleable.GifView, defStyle, R.style.Widget_GifView);
  46. mMovieResourceId = array.getResourceId(R.styleable.GifView_gif, -1);
  47. mPaused = array.getBoolean(R.styleable.GifView_paused, false);
  48. array.recycle();
  49. if (mMovieResourceId != -1) {
  50. mMovie = Movie.decodeStream(getResources().openRawResource(
  51. mMovieResourceId));
  52. }
  53. }
  54. /**
  55. * 设置gif图资源
  56. *
  57. * @param movieResId
  58. */
  59. public void setMovieResource(int movieResId) {
  60. this.mMovieResourceId = movieResId;
  61. mMovie = Movie.decodeStream(getResources().openRawResource(
  62. mMovieResourceId));
  63. requestLayout();
  64. }
  65. public void setMovie(Movie movie) {
  66. this.mMovie = movie;
  67. requestLayout();
  68. }
  69. public Movie getMovie() {
  70. return mMovie;
  71. }
  72. public void setMovieTime(int time) {
  73. mCurrentAnimationTime = time;
  74. invalidate();
  75. }
  76. /**
  77. * 设置暂停
  78. *
  79. * @param paused
  80. */
  81. public void setPaused(boolean paused) {
  82. this.mPaused = paused;
  83. if (!paused) {
  84. mMovieStart = android.os.SystemClock.uptimeMillis()
  85. - mCurrentAnimationTime;
  86. }
  87. invalidate();
  88. }
  89. /**
  90. * 判断gif图是否停止了
  91. *
  92. * @return
  93. */
  94. public boolean isPaused() {
  95. return this.mPaused;
  96. }
  97. @Override
  98. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  99. if (mMovie != null) {
  100. int movieWidth = mMovie.width();
  101. int movieHeight = mMovie.height();
  102. int maximumWidth = MeasureSpec.getSize(widthMeasureSpec);
  103. float scaleW = (float) movieWidth / (float) maximumWidth;
  104. mScale = 1f / scaleW;
  105. mMeasuredMovieWidth = maximumWidth;
  106. mMeasuredMovieHeight = (int) (movieHeight * mScale);
  107. setMeasuredDimension(mMeasuredMovieWidth, mMeasuredMovieHeight);
  108. } else {
  109. setMeasuredDimension(getSuggestedMinimumWidth(),
  110. getSuggestedMinimumHeight());
  111. }
  112. }
  113. @Override
  114. protected void onLayout(boolean changed, int l, int t, int r, int b) {
  115. super.onLayout(changed, l, t, r, b);
  116. mLeft = (getWidth() - mMeasuredMovieWidth) / 2f;
  117. mTop = (getHeight() - mMeasuredMovieHeight) / 2f;
  118. mVisible = getVisibility() == View.VISIBLE;
  119. }
  120. @Override
  121. protected void onDraw(Canvas canvas) {
  122. if (mMovie != null) {
  123. if (!mPaused) {
  124. updateAnimationTime();
  125. drawMovieFrame(canvas);
  126. invalidateView();
  127. } else {
  128. drawMovieFrame(canvas);
  129. }
  130. }
  131. }
  132. @SuppressLint("NewApi")
  133. private void invalidateView() {
  134. if (mVisible) {
  135. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
  136. postInvalidateOnAnimation();
  137. } else {
  138. invalidate();
  139. }
  140. }
  141. }
  142. private void updateAnimationTime() {
  143. long now = android.os.SystemClock.uptimeMillis();
  144. // 如果第一帧,记录起始时间
  145. if (mMovieStart == 0) {
  146. mMovieStart = now;
  147. }
  148. // 取出动画的时长
  149. int dur = mMovie.duration();
  150. if (dur == 0) {
  151. dur = DEFAULT_MOVIE_DURATION;
  152. }
  153. // 算出需要显示第几帧
  154. mCurrentAnimationTime = (int) ((now - mMovieStart) % dur);
  155. }
  156. private void drawMovieFrame(Canvas canvas) {
  157. // 设置要显示的帧,绘制即可
  158. mMovie.setTime(mCurrentAnimationTime);
  159. canvas.save(Canvas.MATRIX_SAVE_FLAG);
  160. canvas.scale(mScale, mScale);
  161. mMovie.draw(canvas, mLeft / mScale, mTop / mScale);
  162. canvas.restore();
  163. }
  164. @SuppressLint("NewApi")
  165. @Override
  166. public void onScreenStateChanged(int screenState) {
  167. super.onScreenStateChanged(screenState);
  168. mVisible = screenState == SCREEN_STATE_ON;
  169. invalidateView();
  170. }
  171. @SuppressLint("NewApi")
  172. @Override
  173. protected void onVisibilityChanged(View changedView, int visibility) {
  174. super.onVisibilityChanged(changedView, visibility);
  175. mVisible = visibility == View.VISIBLE;
  176. invalidateView();
  177. }
  178. @Override
  179. protected void onWindowVisibilityChanged(int visibility) {
  180. super.onWindowVisibilityChanged(visibility);
  181. mVisible = visibility == View.VISIBLE;
  182. invalidateView();
  183. }
  184. }

Movie其实管理着GIF动画中的多个帧,只需要通过 setTime() 一下就可以让它在draw()的时候绘出相应的那帧图像。通过当前时间与duration之间的换算关系,是很容易实现GIF动起来的效果。

其次,在xml布局文件中,把这个view定义进去,代码如下:

[html] view
plain
copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent"
  5. android:orientation="vertical" >
  6. <net.loonggg.gif.view.GifView
  7. android:id="@+id/gif1"
  8. android:layout_width="100dp"
  9. android:layout_height="100dp"
  10. android:layout_gravity="center_horizontal"
  11. android:enabled="false" />
  12. <net.loonggg.gif.view.GifView
  13. android:id="@+id/gif2"
  14. android:layout_width="200dp"
  15. android:layout_height="200dp"
  16. android:layout_gravity="center_horizontal"
  17. android:enabled="false" />
  18. </LinearLayout>

最后,在MainActivity中的使用,代码如下:

[java] view
plain
copy

  1. package net.loonggg.gif;
  2. import net.loonggg.gif.view.GifView;
  3. import android.app.Activity;
  4. import android.os.Bundle;
  5. public class Gif extends Activity {
  6. private GifView gif1, gif2;
  7. @Override
  8. public void onCreate(Bundle savedInstanceState) {
  9. super.onCreate(savedInstanceState);
  10. setContentView(R.layout.main);
  11. gif1 = (GifView) findViewById(R.id.gif1);
  12. // 设置背景gif图片资源
  13. gif1.setMovieResource(R.raw.kitty);
  14. gif2 = (GifView) findViewById(R.id.gif2);
  15. gif2.setMovieResource(R.raw.b);
  16. // 设置暂停
  17. // gif2.setPaused(true);
  18. }
  19. }

注意:与ImageView和其他View唯一的区别在于我加了一个gif属性。

[html] view
plain
copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <resources>
  3. <declare-styleable name="GifView">
  4. <attr name="gif" format="reference" />
  5. <attr name="paused" format="boolean" />
  6. </declare-styleable>
  7. <declare-styleable name="CustomTheme">
  8. <attr name="gifViewStyle" format="reference" />
  9. </declare-styleable>
  10. </resources>

这个代码已经非常好了,使用也非常方便,其实不懂代码是什么意思也可以很好的用,只需要懂得我写注释的那几行和Activity里面的那几行代码就可以了!

转载请注明出处:http://blog.csdn.net/loongggdroid/article/details/21166563

android GifView分享的更多相关文章

  1. android微信分享要注意的地方

    最近在做android端分享的功能,在微信开放平台查看了下官网上的开发文档,一步一步的按文档上的步骤来: 1.申请你的AppID 2.下载开发工具包 3.搭建开发环境,引入libammsdk.jar文 ...

  2. 免费开源的android项目分享

    免费开源的android项目分享:http://yun.baidu.com/share/link?shareid=2945649048&uk=3910054188

  3. fir.im Weekly - 8 个不能错过的 iOS / Android 技术分享

    本期 fir.im Weekly 收集了 2 月下旬新鲜出炉的 iOS /Android 技术分享.源码等,iOS 中图片技术的解压缩.逆向实战.iOS SDK 实践,Android架构思考.Andr ...

  4. Android技术分享-文字转语音并朗读

    Android技术分享-文字转语音并朗读 最近在做一个项目,其中有一个功能是需要将文本转换成语音并播放出来.下面我将我的做法分享一下. 非常令人开心的是,Android系统目前已经集成了TTS,提供了 ...

  5. Android技巧分享——如何用电脑下载在Google play中应用的apk文件

    [Android技巧分享系列] 1.Android技巧分享——让官方模拟器和genymotion虚拟机飞起来 2.Android技巧分享——如何用电脑下载在Google play中应用的apk文件 G ...

  6. Android APP分享功能实现

    [Android应用开发详解]第01期:第三方授权认证(一)实现第三方授权登录.分享以及获取用户资料   由于公司项目的需要,要实现在项目中使用第三方授权登录以及分享文字和图片等这样的效果,几经波折, ...

  7. Android社会化分享功能的实现步骤

    众所周知,互联网是一个资源共享的地方,在网络上,我们可以分享我们所有认为好的资源.而随着互联网信息爆发式的增长,我们习惯了一键分享功能,比如:微博分享.微信分享.QQ空间分享.人人网分享等等.由此可见 ...

  8. Android微信分享功能实例+demo

    Android微信分享功能实例 1 微信开放平台注册 2 获得appId,添加到程序中,并运行程序 3 使用应用签名apk生成签名,添加到微信开放平台应用签名,完成注册 4 测试分享功能. 有问题请留 ...

  9. Android 微信分享与QQ分享功能

    微信分享与QQ分享功能现在都挺常见的,可以根据一些第三方社会化分功能快速实现,不过多多少少都不怎么纯净,最好都是自己看官方文档来实现就最好了~ 一.微信分享 微信分享功能需要先在微信开放平台注册应用并 ...

随机推荐

  1. 2015 多校联赛 ——HDU5373(模拟)

    Problem Description In this problem, we should solve an interesting game. At first, we have an integ ...

  2. 【NOIP2017 OFO(下)】

    ·我不知道对不对,只是不想让大米兔就这样离开.      by tkys_Austin;                    [另一只情绪化的兔子]        今年的11月12日NOIP提高组, ...

  3. 5650 so easy

    so easy  Accepts: 512  Submissions: 1601  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 6553 ...

  4. vim配置文件和插件管理

    本文通过总结零碎的资料总结而成,更多是去引导学习vim配置文件及插件使用. .vimrc配置文件,内容如下(备注清晰) "引入插件pathogen使用 execute pathogen#in ...

  5. 移动端手势双击(MouseDown也可以在移动端响应,但是帧率太低)

    void Update() { if (Input.touchCount > 0)//手指数量 { if(Input.GetTouch(0).phase == TouchPhase.Began ...

  6. Python中模块之copy的功能介绍

    模块之copy的功能介绍 copy主要分两种: 1.浅拷贝 2.深拷贝 赋值: 在python中赋值算特殊的拷贝,其实赋值可以理解为同一个对象有两个名字,所以当其中一个发生变化,另一个也跟着会变化. ...

  7. ZhuSuan 是建立在Tensorflow上的贝叶斯深层学习的 python 库

    ZhuSuan 是建立在Tensorflow上的贝叶斯深层学习的 python 库. 与现有的主要针对监督任务设计的深度学习库不同,ZhuSuan 的特点是深入到贝叶斯推理中,从而支持各种生成模式:传 ...

  8. SQL之排序

    1.按多个列排序 经常需要按不止一个列进行数据排序.例如,如果要显示雇员名单,可能希望按姓和名排序(首先按姓排序,然后在每个姓中再按名排序).如果多个雇员有相同的姓,这样做很有用. 要按多个列排序,简 ...

  9. 1-学习GPRS_Air202(Air202开发板介绍)

    记得自己第一次实现远程通信是在学校里用SIM900A实现的,随着WIFI模块的普及自己就开始用WIFI模块了,当然WIFI模块已经用的很... WIFI模块要想实现远程控制必须连接路由器,其实在做王哥 ...

  10. HTML5 唤起 APP

    <p><a href="xxx://app/question/95">点击跳转,直接回帖报名</a></p> /* global n ...