import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import org.apache.commons.io.IOUtils;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import android.content.Context;
import android.content.res.XmlResourceParser;
import android.graphics.BitmapFactory;
import android.graphics.drawable.AnimationDrawable;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.os.Handler;
import android.widget.ImageView;
/****
* 此工具类源于stack over flow
* 原文链接:http://stackoverflow.com/questions/8692328/causing-outofmemoryerror-in-frame-by-frame-animation-in-android
* 通过底层C来绘制图片,有效防止OOM
* *******/
public class MyAnimationDrawable {
public static class MyFrame {
byte[] bytes;
int duration;
Drawable drawable;
boolean isReady = false;
}
public interface OnDrawableLoadedListener {
public void onDrawableLoaded(List<MyFrame> myFrames);
}
// 1
/***
* 性能更优
* 在animation-list中设置时间
* **/
public static void animateRawManuallyFromXML(int resourceId,
final ImageView imageView, final Runnable onStart,
final Runnable onComplete) {
loadRaw(resourceId, imageView.getContext(),
new OnDrawableLoadedListener() {
@Override
public void onDrawableLoaded(List<MyFrame> myFrames) {
if (onStart != null) {
onStart.run();
}
animateRawManually(myFrames, imageView, onComplete);
}
});
}
// 2
private static void loadRaw(final int resourceId, final Context context,
final OnDrawableLoadedListener onDrawableLoadedListener) {
loadFromXml(resourceId, context, onDrawableLoadedListener);
}
// 3
private static void loadFromXml(final int resourceId,
final Context context,
final OnDrawableLoadedListener onDrawableLoadedListener) {
new Thread(new Runnable() {
@Override
public void run() {
final ArrayList<MyFrame> myFrames = new ArrayList<MyFrame>();
XmlResourceParser parser = context.getResources().getXml(
resourceId);
try {
int eventType = parser.getEventType();
while (eventType != XmlPullParser.END_DOCUMENT) {
if (eventType == XmlPullParser.START_DOCUMENT) {
} else if (eventType == XmlPullParser.START_TAG) {
if (parser.getName().equals("item")) {
byte[] bytes = null;
int duration = 1000;
for (int i = 0; i < parser.getAttributeCount(); i++) {
if (parser.getAttributeName(i).equals(
"drawable")) {
int resId = Integer.parseInt(parser
.getAttributeValue(i)
.substring(1));
bytes = IOUtils.toByteArray(context
.getResources()
.openRawResource(resId));
} else if (parser.getAttributeName(i)
.equals("duration")) {
duration = parser.getAttributeIntValue(
i, 1000);
}
}
MyFrame myFrame = new MyFrame();
myFrame.bytes = bytes;
myFrame.duration = duration;
myFrames.add(myFrame);
}
} else if (eventType == XmlPullParser.END_TAG) {
} else if (eventType == XmlPullParser.TEXT) {
}
eventType = parser.next();
}
} catch (IOException e) {
e.printStackTrace();
} catch (XmlPullParserException e2) {
// TODO: handle exception
e2.printStackTrace();
}
// Run on UI Thread
new Handler(context.getMainLooper()).post(new Runnable() {
@Override
public void run() {
if (onDrawableLoadedListener != null) {
onDrawableLoadedListener.onDrawableLoaded(myFrames);
}
}
});
}
}).run();
}
// 4
private static void animateRawManually(List<MyFrame> myFrames,
ImageView imageView, Runnable onComplete) {
animateRawManually(myFrames, imageView, onComplete, 0);
}
// 5
private static void animateRawManually(final List<MyFrame> myFrames,
final ImageView imageView, final Runnable onComplete,
final int frameNumber) {
final MyFrame thisFrame = myFrames.get(frameNumber);
if (frameNumber == 0) {
thisFrame.drawable = new BitmapDrawable(imageView.getContext()
.getResources(), BitmapFactory.decodeByteArray(
thisFrame.bytes, 0, thisFrame.bytes.length));
} else {
MyFrame previousFrame = myFrames.get(frameNumber - 1);
((BitmapDrawable) previousFrame.drawable).getBitmap().recycle();
previousFrame.drawable = null;
previousFrame.isReady = false;
}
imageView.setImageDrawable(thisFrame.drawable);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
// Make sure ImageView hasn't been changed to a different Image
// in this time
if (imageView.getDrawable() == thisFrame.drawable) {
if (frameNumber + 1 < myFrames.size()) {
MyFrame nextFrame = myFrames.get(frameNumber + 1);
if (nextFrame.isReady) {
// Animate next frame
animateRawManually(myFrames, imageView, onComplete,
frameNumber + 1);
} else {
nextFrame.isReady = true;
}
} else {
if (onComplete != null) {
onComplete.run();
}
}
}
}
}, thisFrame.duration);
// Load next frame
if (frameNumber + 1 < myFrames.size()) {
new Thread(new Runnable() {
@Override
public void run() {
MyFrame nextFrame = myFrames.get(frameNumber + 1);
nextFrame.drawable = new BitmapDrawable(imageView
.getContext().g

sky947635060
- 粉丝: 0
- 资源: 13
最新资源
- 基于最新算法自适应调整模态分解参数的新版麻雀搜索算法优化VMDEMD论文解析,基于模态个数与二次惩罚项优化的麻雀搜索算法自适应变分模态分解参数(SSA VMD)新算法解析及matlab代码实现(适用于
- 基于JavaScript、Vue、CSS和HTML的企业碳足迹与ESG综合管理设计源码
- (源码)基于C语言AVR微控制器的汽车巡航控制系统.zip
- 基于遗传算法的IEEE33节点配电网分布式电源DG优化配置与成本优化研究,利用遗传算法对IEEE 33节点配电网进行分布式电源DG优化配置规划:以总成本最少为目标构建目标函数,涵盖发电成本、网损成本及
- (源码)基于Arduino框架的夜间照明系统.zip
- 基于组态王智能轮胎裂解系统的设计与应用,基于组态王的轮胎裂解系统及其关键技术的研究与探讨,基于组态王轮胎裂解系统 ,基于组态王轮胎裂解系统; 核心关键词为:组态王、轮胎、裂解、系统;,基于组态王的轮胎
- (源码)基于Arduino的隐形手电筒系统.zip
- 基于非对称纳什谈判理论的微网电能共享与P2P交易优化策略:隐私保护下的合作博弈与收益分配机制,基于非对称纳什谈判理论的微网电能共享与P2P交易优化策略:隐私保护下的合作博弈与收益分配,MATLAB代码
- (源码)基于QMK固件库的G844100键盘定制系统.zip
- 基于自抗扰控制的永磁直驱风力发电系统:机侧变流器与网侧变流器的协同优化及风速模型探讨,基于自抗扰控制的永磁直驱风力发电系统模型设计与优化研究:LADRC应用于机侧变流器与五种风速模型下的最大功率跟踪研
- (源码)基于STM32F10x框架的PS2手柄驱动程序.zip
- 基于S7-200 PLC和组态王动画仿真的水箱水位智能控制系统设计与实现:附PLC源代码详解、IO地址分配及图纸,基于S7-200 PLC与组态王动画仿真的水箱水位智能控制系统设计,含PLC与组态王源
- (源码)基于Python框架的智能微信机器人.zip
- 基于MPC的无人驾驶轨迹重规划跟踪技术:在Carsim与Simulink中自定义障碍物的应用,基于MPC的无人驾驶轨迹重规划跟踪技术:在Carsim与Simulink中自定义障碍物的应用,自定义障碍物
- 基于Python+Selenium的WebUI自动化测试框架设计源码
- (源码)基于物联网和Arduino的安全监控系统.zip
资源上传下载、课程学习等过程中有任何疑问或建议,欢迎提出宝贵意见哦~我们会及时处理!
点击此处反馈


