`
砺雪凝霜
  • 浏览: 157278 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

开源框架ListViewAnimations解析

阅读更多

 

1 概念

ListViewAnimations是一个带Item显示动画的ListView,动画包括底部飞入、其他方向斜飞入、下层飞入、渐变消失、滑动删除等

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/
 

 

 
 
 
 
 
 
 
 
 
 
 
 
 
3
0
分享到:
评论

相关推荐

    google开源框架ListViewAnimations

    《深入解析Google开源框架ListViewAnimations》 在Android开发中,ListView作为一款常用的数据展示控件,经常被用于显示大量的列表数据。然而,为了让用户界面更加生动有趣,开发者们常常会寻求给ListView添加动画...

    Android 开源框架源码解读.txt

    第5章 LeakCanary内存泄漏框架解析和相关面试题分析 第6章 butterknife依赖注入框架源码解析 第8章 eventbus异步框架源码解析 第9章 dagger2依赖注入框架源码解析 第10章 rxjava异步框架源码解析 第11章 ...

    renren-fast开源框架开发文档2.0_完整版.zip

    《人人快开框架开发文档2.0:SpringBoot+Vue深度解析》 “人人快开”(Renren-Fast)是一款基于SpringBoot和Vue.js的开源框架,它为开发者提供了快速构建企业级Web应用的能力。这份《Renren-Fast开源框架开发文档...

    JavaEE主流开源框架-Struts部分rmvb格式.zip

    JavaEE主流开源框架-Struts部分rmvb格式. JavaEE主流开源框架-Struts部分rmvb格式. JavaEE主流开源框架-Struts部分rmvb格式. JavaEE主流开源框架-Struts部分rmvb格式. JavaEE主流开源框架-Struts部分rmvb格式. ...

    BAT大牛带你深度剖析Android 10大开源框架

    BAT大牛 带你深度剖析Android 10大开源框架 好东西分享学习进步

    无刷新整合型网站开源框架

    无刷新整合型网站开源框架是一种先进的网页开发技术,它允许用户在不重新加载整个页面的情况下更新部分网页内容,显著提升了用户体验。这种技术的核心是利用Ajax(Asynchronous JavaScript and XML)异步通信,配合...

    基于Devexpress的winform开源框架(伍老师de )

    基于Devexpress的winform开源框架,包括源码。 网盘密码: a123

    WPF MVVM 开源框架,包含数据验证,很不错的

    **WPF MVVM 开源框架概述** WPF(Windows Presentation Foundation)是微软.NET Framework的一部分,用于构建具有丰富用户界面的应用程序。MVVM(Model-View-ViewModel)是一种设计模式,常用于WPF开发中,它将业务...

    基于Cad二次开发的开源框架NFox

    《基于Cad二次开发的开源框架NFox深度解析》 在当今的计算机辅助设计(CAD)领域,二次开发已经成为提升效率、实现个性化定制的关键途径。本文将深入探讨一款基于.NET Framework和C#语言的开源框架——NFox,它为...

    开源框架license整理

    在IT行业中,开源框架扮演着至关重要的角色,它们为开发者提供了快速构建应用程序的基础,极大地提高了开发效率。开源框架的使用通常伴随着一个关键因素——license,它定义了用户可以如何使用、修改和分发这些框架...

    计算机毕业毕业设计成品 JdonFramework开源框架 v5.1 Build20071025-jdonframework

    计算机毕业毕业设计成品 JdonFramework开源框架 v5.1 Build20071025_jdonframework- 计算机毕业毕业设计成品 JdonFramework开源框架 v5.1 Build20071025_jdonframework- 计算机毕业毕业设计成品 JdonFramework开源...

    .net开源框架

    .NET开源框架是软件开发领域中的一个重要组成部分,它为开发者提供了构建高效、稳定和可扩展的应用程序的基础设施。本文将深入探讨.NET开源框架的核心概念、功能特性以及如何利用它来加速开发进程。 首先,.NET框架...

    基于Unity开源框架GameFramewrk实现的一款塔防游戏Demo源码+项目说明.zip

    【资源说明】 1、该资源包括项目的全部源码,下载可以直接使用! 2、本项目适合作为计算机、数学、电子信息等专业的课程设计、期末大作业和...基于Unity开源框架GameFramewrk实现的一款塔防游戏Demo源码+项目说明.zip

    基于Devexpress的winform开源框架,带源码

    ### 基于Devexpress的WinForm开源框架知识点解析 #### 一、DevExpress与WinForms简介 DevExpress是一家专注于为.NET平台提供高质量控件组件的软件公司。DevExpress的产品因其丰富的功能、出色的性能以及良好的用户...

    Android第三方开源框架ImageLoader的完美Demo

    `ImageLoader`就是这样一个被广泛使用的第三方开源框架,它为开发者提供了强大的图片异步加载和缓存功能。本Demo详细展示了如何在Android项目中集成和使用`ImageLoader`,以实现高效、流畅的图片显示。 `...

    Android afinal开源框架实例源码.zip项目安卓应用源码下载

    Android afinal开源框架实例源码.zip项目安卓应用源码下载Android afinal开源框架实例源码.zip项目安卓应用源码下载 1.适合学生毕业设计研究参考 2.适合个人学习研究参考 3.适合公司开发项目技术参考

    beetle 开源框架书的源码

    【标题】"beetle 开源框架书的源码"涉及的是一个名为"beetle"的开源框架的源代码,这通常意味着该框架可能是用于Java平台,因为J2EE(Java 2 Platform, Enterprise Edition)是Java领域广泛使用的开发企业级应用的...

    人人开源框架renren-security-master(vue2版)

    人人开源框架renren-security-master(vue2版)

    一个AT指令解析框架,mark一下

    AT指令解析框架是一种用于处理和解析AT(Attention)指令的软件工具,广泛应用于通信设备,如GSM/GPRS模块、蓝牙模块、Wi-Fi模块等。AT指令是串行通信中的控制命令,允许用户通过简单的文本命令与硬件进行交互,设置...

Global site tag (gtag.js) - Google Analytics