服务器之家

服务器之家 > 正文

Android中显示GIF动画的实现代码

时间:2021-04-05 15:07     来源/作者:非著名程序员

本文实例讲述了android中显示gif动画的实现代码。分享给大家供大家参考,具体如下:

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

废话不多说,直接上图:

Android中显示GIF动画的实现代码

Android中显示GIF动画的实现代码

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
package net.loonggg.gif.view;
import net.loonggg.gif.r;
import android.annotation.suppresslint;
import android.content.context;
import android.content.res.typedarray;
import android.graphics.canvas;
import android.graphics.movie;
import android.os.build;
import android.util.attributeset;
import android.view.view;
public class gifview extends view {
  /**
   * 默认为1秒
   */
  private static final int default_movie_duration = 1000;
  private int mmovieresourceid;
  private movie mmovie;
  private long mmoviestart;
  private int mcurrentanimationtime = 0;
  private float mleft;
  private float mtop;
  private float mscale;
  private int mmeasuredmoviewidth;
  private int mmeasuredmovieheight;
  private boolean mvisible = true;
  private volatile boolean mpaused = false;
  public gifview(context context) {
    this(context, null);
  }
  public gifview(context context, attributeset attrs) {
    this(context, attrs, r.styleable.customtheme_gifviewstyle);
  }
  public gifview(context context, attributeset attrs, int defstyle) {
    super(context, attrs, defstyle);
    setviewattributes(context, attrs, defstyle);
  }
  @suppresslint("newapi")
  private void setviewattributes(context context, attributeset attrs,
      int defstyle) {
    if (build.version.sdk_int >= build.version_codes.honeycomb) {
      setlayertype(view.layer_type_software, null);
    }
    // 从描述文件中读出gif的值,创建出movie实例
    final typedarray array = context.obtainstyledattributes(attrs,
        r.styleable.gifview, defstyle, r.style.widget_gifview);
    mmovieresourceid = array.getresourceid(r.styleable.gifview_gif, -1);
    mpaused = array.getboolean(r.styleable.gifview_paused, false);
    array.recycle();
    if (mmovieresourceid != -1) {
      mmovie = movie.decodestream(getresources().openrawresource(
          mmovieresourceid));
    }
  }
  /**
   * 设置gif图资源
   *
   * @param movieresid
   */
  public void setmovieresource(int movieresid) {
    this.mmovieresourceid = movieresid;
    mmovie = movie.decodestream(getresources().openrawresource(
        mmovieresourceid));
    requestlayout();
  }
  public void setmovie(movie movie) {
    this.mmovie = movie;
    requestlayout();
  }
  public movie getmovie() {
    return mmovie;
  }
  public void setmovietime(int time) {
    mcurrentanimationtime = time;
    invalidate();
  }
  /**
   * 设置暂停
   *
   * @param paused
   */
  public void setpaused(boolean paused) {
    this.mpaused = paused;
    if (!paused) {
      mmoviestart = android.os.systemclock.uptimemillis()
          - mcurrentanimationtime;
    }
    invalidate();
  }
  /**
   * 判断gif图是否停止了
   *
   * @return
   */
  public boolean ispaused() {
    return this.mpaused;
  }
  @override
  protected void onmeasure(int widthmeasurespec, int heightmeasurespec) {
    if (mmovie != null) {
      int moviewidth = mmovie.width();
      int movieheight = mmovie.height();
      int maximumwidth = measurespec.getsize(widthmeasurespec);
      float scalew = (float) moviewidth / (float) maximumwidth;
      mscale = 1f / scalew;
      mmeasuredmoviewidth = maximumwidth;
      mmeasuredmovieheight = (int) (movieheight * mscale);
      setmeasureddimension(mmeasuredmoviewidth, mmeasuredmovieheight);
    } else {
      setmeasureddimension(getsuggestedminimumwidth(),
          getsuggestedminimumheight());
    }
  }
  @override
  protected void onlayout(boolean changed, int l, int t, int r, int b) {
    super.onlayout(changed, l, t, r, b);
    mleft = (getwidth() - mmeasuredmoviewidth) / 2f;
    mtop = (getheight() - mmeasuredmovieheight) / 2f;
    mvisible = getvisibility() == view.visible;
  }
  @override
  protected void ondraw(canvas canvas) {
    if (mmovie != null) {
      if (!mpaused) {
        updateanimationtime();
        drawmovieframe(canvas);
        invalidateview();
      } else {
        drawmovieframe(canvas);
      }
    }
  }
  @suppresslint("newapi")
  private void invalidateview() {
    if (mvisible) {
      if (build.version.sdk_int >= build.version_codes.jelly_bean) {
        postinvalidateonanimation();
      } else {
        invalidate();
      }
    }
  }
  private void updateanimationtime() {
    long now = android.os.systemclock.uptimemillis();
    // 如果第一帧,记录起始时间
    if (mmoviestart == 0) {
      mmoviestart = now;
    }
    // 取出动画的时长
    int dur = mmovie.duration();
    if (dur == 0) {
      dur = default_movie_duration;
    }
    // 算出需要显示第几帧
    mcurrentanimationtime = (int) ((now - mmoviestart) % dur);
  }
  private void drawmovieframe(canvas canvas) {
    // 设置要显示的帧,绘制即可
    mmovie.settime(mcurrentanimationtime);
    canvas.save(canvas.matrix_save_flag);
    canvas.scale(mscale, mscale);
    mmovie.draw(canvas, mleft / mscale, mtop / mscale);
    canvas.restore();
  }
  @suppresslint("newapi")
  @override
  public void onscreenstatechanged(int screenstate) {
    super.onscreenstatechanged(screenstate);
    mvisible = screenstate == screen_state_on;
    invalidateview();
  }
  @suppresslint("newapi")
  @override
  protected void onvisibilitychanged(view changedview, int visibility) {
    super.onvisibilitychanged(changedview, visibility);
    mvisible = visibility == view.visible;
    invalidateview();
  }
  @override
  protected void onwindowvisibilitychanged(int visibility) {
    super.onwindowvisibilitychanged(visibility);
    mvisible = visibility == view.visible;
    invalidateview();
  }
}

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

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

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
package net.loonggg.gif;
import net.loonggg.gif.view.gifview;
import android.app.activity;
import android.os.bundle;
public class gif extends activity {
  private gifview gif1, gif2;
  @override
  public void oncreate(bundle savedinstancestate) {
    super.oncreate(savedinstancestate);
    setcontentview(r.layout.main);
    gif1 = (gifview) findviewbyid(r.id.gif1);
    // 设置背景gif图片资源
    gif1.setmovieresource(r.raw.kitty);
    gif2 = (gifview) findviewbyid(r.id.gif2);
    gif2.setmovieresource(r.raw.b);
    // 设置暂停
    // gif2.setpaused(true);
  }
}

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

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

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

希望本文所述对大家android程序设计有所帮助。

标签:

相关文章

热门资讯

2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全
2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全 2019-12-26
yue是什么意思 网络流行语yue了是什么梗
yue是什么意思 网络流行语yue了是什么梗 2020-10-11
背刺什么意思 网络词语背刺是什么梗
背刺什么意思 网络词语背刺是什么梗 2020-05-22
Intellij idea2020永久破解,亲测可用!!!
Intellij idea2020永久破解,亲测可用!!! 2020-07-29
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总 2020-11-13
返回顶部