1 概念
ListViewAnimations是一个带Item显示动画的ListView,动画包括底部飞入、其他方向斜飞入、下层飞入、渐变消失、滑动删除等
项目地址:https://github.com/nhaarman/ListViewAnimations
Demo地址:https://play.google.com/store/apps/details?id=com.haarman.listviewanimations
Demo地址:https://play.google.com/store/apps/details?id=com.haarman.listviewanimations
2 使用
(1) 添加一下库文件
- lib-core:这是ListViewAnimations的核心库,它包含各种各样的动画效果
- lib-manipulation:包括一些对listView item的操作,例如 Swipe-to-Dismiss, and Drag-and-Drop
- lib-core=slh:对核心库库进行了扩展,支持StickyListHeaders(轻松给listView添加header)
(2) build.gradle配置
repositories {
mavenCentral()
}
dependencies {
compile 'com.nhaarman.listviewanimations:lib-core:3.1.0@aar'
compile 'com.nhaarman.listviewanimations:lib-manipulation:3.1.0@aar'
compile 'com.nhaarman.listviewanimations:lib-core-slh:3.1.0@aar'
}
3 如何移植项目
在导入了上述三个库文件的前提下
(1) 初始化ListView
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_appearanceexample);
mListView = (ListView) findViewById(R.id.activity_appearanceexample_listview);
mCurrentListView = mListView;
mAdapter = new MyAdapter(this, getItems());
setRightAdapter();
}
(2)调用ListViewAnimations提供的动画API
private void setRightAdapter() {
AnimationAdapter animAdapter = new SwingRightInAnimationAdapter(mAdapter);
animAdapter.setAbsListView(mCurrentListView);
mCurrentListView.setAdapter(animAdapter);
}
4 ListViewAnimations源码解析
我就拿一个从listView item从右边慢慢显示出来,期间前一个item比后一个item先出现的动画效果
4.1构造一个SwingRightInAnimationAdapter对象
我们使用SwingRightInAnimationAdapter的时候你可以看到以下代码:
protected final BaseAdapter mDecoratedBaseAdapter;
private AbsListView mListView;
public BaseAdapterDecorator(BaseAdapter baseAdapter) {
mDecoratedBaseAdapter = baseAdapter;
}
public void setAbsListView(AbsListView listView) {
mListView = listView;
if (mDecoratedBaseAdapter instanceof BaseAdapterDecorator) {
((BaseAdapterDecorator) mDecoratedBaseAdapter).setAbsListView(listView);
}
if (mListView instanceof DynamicListView) {
DynamicListView dynListView = (DynamicListView) mListView;
dynListView.setIsParentHorizontalScrollContainer(mIsParentHorizontalScrollContainer);
dynListView.setDynamicTouchChild(mResIdTouchChild);
}
}
构造一个SwingRightInAnimationAdapter类的时候传入了一个BaseAdapter对象,同时也设置了一个ListView对象也作为它的成员类,这个adapter和listView就是我们项目中原来就有的adapter和ListView,同事它还复写了adapter中所有的方法,例如getView,getItem,getViewType...
这用到了装饰设计模式,在不改变原来类的情况下,对原来的类进行增强。典型的OCP原则,一个好的程序是对内封闭,对外开放的。
4.2在来看看SwingRightInAnimationAdapter的源码
SwingRightInAnimationAdapter是抽象类SingleAnimationAdapter 的子类,它继承了getAnimator()方法
public class SwingRightInAnimationAdapter extends SingleAnimationAdapter {
private final long mAnimationDelayMillis;
private final long mAnimationDurationMillis;
public SwingRightInAnimationAdapter(BaseAdapter baseAdapter) {
this(baseAdapter, DEFAULTANIMATIONDELAYMILLIS, DEFAULTANIMATIONDURATIONMILLIS);
}
public SwingRightInAnimationAdapter(BaseAdapter baseAdapter, long animationDelayMillis) {
this(baseAdapter, animationDelayMillis, DEFAULTANIMATIONDURATIONMILLIS);
}
public SwingRightInAnimationAdapter(BaseAdapter baseAdapter, long animationDelayMillis, long animationDurationMillis) {
super(baseAdapter);
mAnimationDelayMillis = animationDelayMillis;
mAnimationDurationMillis = animationDurationMillis;
}
@Override
protected long getAnimationDelayMillis() {
return mAnimationDelayMillis;
}
@Override
protected long getAnimationDurationMillis() {
return mAnimationDurationMillis;
}
@Override
protected Animator getAnimator(ViewGroup parent, View view) {
return ObjectAnimator.ofFloat(view, "translationX", parent.getWidth(), 0);
}
}
其中:
getAnimationDelayMillis:是获取每一个item动画延迟加载的时间
getAnimationDurationMillis:动画运行时间
getAnimator:动画的类型(item的宽度从0渐变到listView的宽度的一个动画),不同类型的动画该方法的实现并不相同。
4.3 AnimationAdapter源码
AnimationAdapter是BaseAdapterDecorator的子类,在4.1中我们已经知道了,BaseAdapterDecorator中有getView,getItem这些类似于adapter的方法其实就是我们自己定义的ListView的适配器中的方法.
@Override
public int getCount() {
return mDecoratedBaseAdapter.getCount();
}
@Override
public Object getItem(int position) {
return mDecoratedBaseAdapter.getItem(position);
}
@Override
public long getItemId(int position) {
return mDecoratedBaseAdapter.getItemId(position);
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
return mDecoratedBaseAdapter.getView(position, convertView, parent);
}
而在AnimationAdapter中我们发现它重写了AnimationAdapter的getView方法(这其实就是我们的adapter的getView方法)
@Override
public final View getView(int position, View convertView, ViewGroup parent) {
boolean alreadyStarted = false;
if (!mHasParentAnimationAdapter) {
if (getAbsListView() == null) {
throw new IllegalStateException("Call setListView() on this AnimationAdapter before setAdapter()!");
}
if (convertView != null) {
alreadyStarted = cancelExistingAnimation(position, convertView);
}
}
View itemView = super.getView(position, convertView, parent);
if (!mHasParentAnimationAdapter && !alreadyStarted) {
animateViewIfNecessary(position, itemView, parent);
}
return itemView;
}
这段代码是实现ListView item 从右往左出现动画的核心代码,思路是:
(1) 由于getView是瞬时的,为了避免item瞬时的显示出来,先要调用hideView()方法隐藏item
private void hideView(View view) {
ObjectAnimator animator = ObjectAnimator.ofFloat(view, "alpha", 0);
AnimatorSet set = new AnimatorSet();
set.play(animator);
set.setDuration(0);
set.start();
}
(2) 拿到item从往左显示的动画,这个动画是实质上就是SwingRightInAnimationAdapter中的getAnimator()方法
@Override
protected Animator getAnimator(ViewGroup parent, View view) {
return ObjectAnimator.ofFloat(view, "translationX", parent.getWidth(), 0);
}
(3) 设置item动画延迟时间
item从右往左显示是一个T字形出来的,所以每个position上的item的动画的延迟时间是不一样的。
set.setStartDelay(calculateAnimationDelay(isHeader));
我们看看calculateAnimationDelay方法是怎么写的
@SuppressLint("NewApi")
private long calculateAnimationDelay(boolean isHeader) {
long delay;
int numberOfItems = getAbsListView().getLastVisiblePosition() - getAbsListView().getFirstVisiblePosition();
if (numberOfItems + 1 < mLastAnimatedPosition) {
delay = getAnimationDelayMillis();
if (getAbsListView() instanceof GridView && Build.VERSION.SDK_INT >= 11) {
delay += getAnimationDelayMillis() * ((mLastAnimatedPosition + 1) % ((GridView) getAbsListView()).getNumColumns());
}
} else {
long delaySinceStart = (mLastAnimatedPosition - mFirstAnimatedPosition + 1) * getAnimationDelayMillis();
delay = mAnimationStartMillis + getInitialDelayMillis() + delaySinceStart - System.currentTimeMillis();
delay -= isHeader && mLastAnimatedPosition > 0 ? getAnimationDelayMillis() : 0;
}
// System.out.println(isHeader + ": " + delay);
return Math.max(0, delay);
}
(4)开始动画集合
Animator[] animators = getAnimators(parent, view);
Animator alphaAnimator = ObjectAnimator.ofFloat(view, "alpha", 0, 1);
AnimatorSet set = new AnimatorSet();
set.playTogether(concatAnimators(childAnimators, animators, alphaAnimator));
set.setStartDelay(calculateAnimationDelay(isHeader));
set.setDuration(getAnimationDurationMillis());
set.start();
源码就这么简单,原理就是当adapter再调用getView的时候,给每个item一个动画效果。
5 自定义ListView item从右往左出现的动画效果
ListViewAnimations看上去天衣无缝,但是当我们仅仅需要在点击一个button的时候,出现这个动画效果,你会发现它满足不了需要,这个时候就要自己写一个ListViewAnimation了。
看看我们项目中的需求:
点击“换一批”要让Item从右往左呈阶梯形出现。但是不希望listView滑动的时候不要出现这种效果,很明显我们要改ListViewAnimations
5.1 新建一个IAnimationManager接口,定义一个从右往左出现动画
public interface IAnimationManager {
public void startRightAppearanceAnimation(int position,View view, ViewGroup parent);
}
定义接口的原因是扩展用,以后要是产品经理说动画不要了,你换别的。这个时候我原来的代码基本不要动,只要在接口中增加一个方法即可。
5.2 新建一个类实现IAnimationManager接口
public class AnimationModel implements IAnimationManager {
private boolean ifCanAnimation;//是否启动动画,true运行动画,false动画停止
/**
* listView item从右边出现动画效果
*/
@Override
public void startRightAppearanceAnimation(int position,View view, ViewGroup parent) {
if(!ifCanAnimation) return;
//隐藏itemView
ObjectAnimator animator = ObjectAnimator.ofFloat(view, "alpha", 0);
AnimatorSet hiddenSet = new AnimatorSet();
hiddenSet.play(animator);
hiddenSet.setDuration(0);
hiddenSet.start();
//item从右出现动画
view.measure(0,0);
ObjectAnimator oa = ObjectAnimator.ofFloat(view, "translationX", view.getMeasuredWidth(), 0);
Animator alphaAnimator = ObjectAnimator.ofFloat(view, "alpha", 0, 1);
AnimatorSet set = new AnimatorSet();
long delay = 100 * (position-1);
set.playTogether(oa,alphaAnimator);
set.setStartDelay(delay);
set.start();
}
public boolean getIfCanAnimation() {
return ifCanAnimation;
}
public void setIfCanAnimation(boolean ifCanAnimation) {
this.ifCanAnimation = ifCanAnimation;
}
}
(1)重写StartRightAppearanceAnimation()方法,并定义一个标志位来控制动画是否能开启
(2)隐藏Item
ObjectAnimator animator = ObjectAnimator.ofFloat(view, "alpha", 0);
AnimatorSet hiddenSet = new AnimatorSet();
hiddenSet.play(animator);
hiddenSet.setDuration(0);
hiddenSet.start();
(3)item出现动画
view.measure(0,0);
ObjectAnimator oa = ObjectAnimator.ofFloat(view, "translationX", view.getMeasuredWidth(), 0);
Animator alphaAnimator = ObjectAnimator.ofFloat(view, "alpha", 0, 1);
AnimatorSet set = new AnimatorSet();
long delay = 100 * (position-1);
set.playTogether(oa,alphaAnimator);
set.setStartDelay(delay);
set.start();
(4) 在adapter中定义一个动画开启方法
/**
* 换一批动画效果
* @param position
* @param view
* @param parent
*/
private void startAnimations(int position, View view, ViewGroup parent){
animationModel.startRightAppearanceAnimation(position,view,parent);
if(getItemSize() == position){
animationModel.setIfCanAnimation(false);
}
}
当滑动到最后一个item的时候,设置不能动画不能运行
(5) 点击"换一批"按钮的时候,设置动画不能运行
animationModel.setIfCanAnimation(true);
说明:我这里使用的ObjectAnimator是android自带的API,需要API15才支持,所以在低版本的手机上是看不到动画效果的,这个时候你可以导入一个nineoldandroids-2.4.0jar,就可以兼容3.0以上的系统。
nineoldandroids-2.4.0jar 下载地址:http://nineoldandroids.com/
相关推荐
第5章 LeakCanary内存泄漏框架解析和相关面试题分析 第6章 butterknife依赖注入框架源码解析 第8章 eventbus异步框架源码解析 第9章 dagger2依赖注入框架源码解析 第10章 rxjava异步框架源码解析 第11章 ...
"第三方开源框架"这一主题涵盖了广泛的知识点,包括但不限于框架的选择、使用方法、许可证合规性、社区支持以及集成到项目中的最佳实践。 首先,我们来理解什么是开源框架。开源框架是一种软件开发工具,它提供了...
《人人快开框架开发文档2.0:SpringBoot+Vue深度解析》 “人人快开”(Renren-Fast)是一款基于SpringBoot和Vue.js的开源框架,它为开发者提供了快速构建企业级Web应用的能力。这份《Renren-Fast开源框架开发文档...
BAT大牛 带你深度剖析Android 10大开源框架 好东西分享学习进步
基于Devexpress的winform开源框架,包括源码。 网盘密码: a123
《基于Cad二次开发的开源框架NFox深度解析》 在当今的计算机辅助设计(CAD)领域,二次开发已经成为提升效率、实现个性化定制的关键途径。本文将深入探讨一款基于.NET Framework和C#语言的开源框架——NFox,它为...
**WPF MVVM 开源框架概述** WPF(Windows Presentation Foundation)是微软.NET Framework的一部分,用于构建具有丰富用户界面的应用程序。MVVM(Model-View-ViewModel)是一种设计模式,常用于WPF开发中,它将业务...
springCloud是基于SpringBoot的一整套实现微服务的框架。他提供了微服务开发所需的配置管理、服务发现、断路器、智能路由、微代理、控制总线、全局锁、决策竞选、分布式会话和集群状态管理等组件。最重要的是, 跟...
计算机毕业毕业设计成品 JdonFramework开源框架 v5.1Build20071025_jdonframework- 计算机毕业毕业设计成品 JdonFramework开源框架 v5.1Build20071025_jdonframework- 计算机毕业毕业设计成品 JdonFramework开源框架...
【58同城开源框架】是58同城公司推出的一款开源技术框架,旨在为开发者提供高效、稳定、可扩展的开发工具。这个框架凝聚了58同城在互联网服务领域的技术积累,体现了其对软件工程的最佳实践,有助于提升开发效率,...
【资源说明】 1、该资源包括项目的全部源码,下载可以直接使用! 2、本项目适合作为计算机、数学、电子信息等专业的课程设计、期末大作业和...基于Unity开源框架GameFramewrk实现的一款塔防游戏Demo源码+项目说明.zip
### 基于Devexpress的WinForm开源框架知识点解析 #### 一、DevExpress与WinForms简介 DevExpress是一家专注于为.NET平台提供高质量控件组件的软件公司。DevExpress的产品因其丰富的功能、出色的性能以及良好的用户...
`ImageLoader`就是这样一个被广泛使用的第三方开源框架,它为开发者提供了强大的图片异步加载和缓存功能。本Demo详细展示了如何在Android项目中集成和使用`ImageLoader`,以实现高效、流畅的图片显示。 `...
JavaEE主流开源框架-Hibernate,视频文件,供学习使用
Ext框架JavaScript开源框架Ext框架JavaScript开源框架
JdonFramework开源框架JdonFramework开源框架JdonFramework开源框架JdonFramework开源框架JdonFramework开源框架JdonFramework开源框架
java开源框架集java开源框架集java开源框架集java开源框架集
人人开源框架renren-security-master(vue2版)
本课程将带你深度剖析Android主流开源框架的源码,让你全面掌握框架的使用场景、内部机制、构造原理、核心类、架构与设计思想等,提升你的代码阅读与分析能力、提高代码设计能力及改造能力,快速突破技术瓶颈,轻松...
C#开源框架是一种基于.NET Framework或.NET Core的高效开发工具,专为简化和加速软件项目的构建而设计。这种框架通常包含一系列预构建的类库、模板和工具,可以帮助开发者快速实现常见的业务逻辑和UI功能,从而提高...