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/
相关推荐
《深入解析Google开源框架ListViewAnimations》 在Android开发中,ListView作为一款常用的数据展示控件,经常被用于显示大量的列表数据。然而,为了让用户界面更加生动有趣,开发者们常常会寻求给ListView添加动画...
Gson: 完成json的解析 Android-PullToRefresh : 下拉刷新和到底部自动加载,以后可能换成SwipeRefreshLayout butterknife: view注入框架,简化开发,减少代码。 ListViewAnimations:ListView添加动画 ...
1 PartA2_SD Host_Controller_Simplified_Specification_Ver4.20 2 PartA2_SD_Host_Controller_Simplified_Specification_Ver2.00 3 PartE1_SDIO_Simplified_Specification_Ver2.00 4 PartE1_SDIO_Simplified_Specification_Ver3.00 5 Part1 PhysicalLayerSimplifiedSpecificationVer9.10Fin_20231201 6 PartE7_Wireless_LAN_Simplified_Addendum_Ver1.10 7 Part1_Extended_Security_Simplified_Addendum_Ver1.00 8 Part1_NFC_Interface_Simplified_Addendum_Ver1.00 9 Part1_UHS-II_Simplified_Addendum_Ver1.02 10 PartA1_ASSD_Extension_Simplified_Specification_Ver2.00 11 PartE2_SDIO Bluetooth_Type_A_Simplified_Specification_Ver1.00 12 SDUC-Host-Implementation-Guideline_Ver1.00
《步入元宇宙》由马克·范·里门撰写,是一本深入探讨元宇宙概念、历史、现状以及未来潜力的书籍。作者从Web 1.0到Web 3.0的发展讲起,详细分析了从增强现实(AR)到虚拟现实(VR)再到扩展现实(XR)的技术演进。书中提出了元宇宙的六大特征:互操作性、去中心化、持久性、空间性、社区驱动和自我主权,并强调了开放元宇宙的重要性及其带来的自由和创新潜力。作者还探讨了元宇宙对个人身份、商业、教育、娱乐等领域的深远影响,并预测了元宇宙将如何推动形成一个全新的社交经济。书中引用了多位行业专家的评价,强调了无论读者对元宇宙的了解程度如何,都能从中获得新的见解和启发。
卢益峰ads仿真放大器章节所需的ads库和MW6S004的ads模型
javaSE阶段面试题
《网页制作基础教程(Dreamweaver-CS6版)》第10章-网站的管理与上传.pptx
内容概要:本文详细介绍了如何使用Abaqus软件构建双线盾构隧道的超精细模型,特别是针对隧道间的联络通道、软化模量和盾构注浆等关键要素进行了深入探讨。文章首先阐述了模型的整体架构搭建,包括使用Python脚本创建隧道衬砌部件。接下来,讨论了软化模量的引入及其在材料本构模型中的定义方式,展示了如何通过塑性应变来模拟软化模量的变化。此外,文章详细讲解了盾构注浆的模拟方法,如通过单元生死技术激活注浆体单元,并提供了具体的Python代码示例。最后,文章强调了网格划分、接触设置等方面的注意事项,确保模型能够稳定运行并获得精确的结果。 适合人群:从事隧道工程数值模拟的研究人员和技术人员,尤其是熟悉Abaqus软件的工程师。 使用场景及目标:适用于需要进行双线盾构隧道工程力学行为研究的场合,旨在帮助工程师更好地理解和预测隧道施工过程中可能出现的问题,从而优化设计方案,提高施工效率和安全性。 其他说明:文中提供的代码片段和建模技巧基于作者的实际经验和测试结果,对于初学者而言,建议逐步尝试每个步骤并在实践中不断调整参数以适应具体工程项目的需求。
《自然资源信息化时代背景与发展》.pdf
《网络社会学(第2版)》15-网络社会变迁.ppt
内容概要:本文详细介绍了使用西门子1214PLC和KTP700Basic PN触摸屏构建双相机四轴多工位检测设备的具体实现方法。主要内容涵盖硬件配置、程序主体功能及其代码解析、触摸屏功能实现等方面。硬件方面,采用了西门子1214PLC作为核心控制器,KTP700Basic PN触摸屏为人机界面,双相机用于检测,第三设备通过Modbus RTU通讯。程序主体功能包括上下双工位4轴脉冲控制步进电机、与上位机双相机的TCP/IP通讯、与第三设备的Modbus RTU通讯。触摸屏功能则涉及多重画面、配方管理和密码保护等功能。文中还分享了一些调试经验和注意事项,如轴使能信号要用上升沿触发、相机通讯需配置心跳包机制等。 适合人群:从事工业自动化领域的工程师和技术人员,特别是那些对PLC编程、触摸屏应用和多工位检测设备感兴趣的读者。 使用场景及目标:适用于需要构建复杂自动化检测系统的工程项目,旨在提高检测效率和准确性,确保设备稳定可靠运行。通过学习本文,读者能够掌握如何使用西门子1214PLC和KTP700触摸屏搭建类似的检测系统。 其他说明:文中提供了大量具体的代码示例和调试技巧,有助于读者更好地理解和实施相关技术。此外,还强调了实际工程中常见的问题及解决方案,如接线和接地问题、通讯参数配置等。
- **4.4 版本** - 介绍了基础特性和标准,适合初学者了解eMMC的基本框架。 - **4.41 版本** - 对4.4版进行了修订和完善,优化了部分规范以适应市场和技术的发展。 - **4.5 版本** - 引入了新的性能改进和技术特性,进一步提升了存储效率。 - **4.51 版本** - 包含针对4.5版的小幅修正和增强,确保技术规范的准确性和实用性。 - **5.0 版本** - 重大更新,引入更多高级功能,支持更高的数据传输速率,对现代高性能需求进行了响应。 - **5.01 版本** - 在5.0基础上的维护更新,保持标准的一致性和先进性。 - **5.1 版本** - 最新的公开版本之一,提供了更全面的标准规范,加强了数据管理能力,提升了可靠性
DeepSeek系列-提示词工程和落地场景.pdf
JDK(java)安装及配置
内容概要:本文详细介绍了引力搜索算法(Gravitational Search Algorithm, GSA)的原理、MATLAB实现及其应用场景。首先解释了GSA的基本概念,即将优化问题中的候选解视为宇宙中互相吸引的粒子,通过模拟物理现象进行优化。接着展示了核心的粒子运动方程,包括加速度计算、质量分配以及引力公式的具体实现。文中提供了多个经典的测试函数如Sphere、Rastrigin等用于验证算法性能,并通过动态绘图展示了粒子群的收敛过程。此外,讨论了算法参数设置的影响,如引力常数G的指数衰减方式,以及如何通过添加随机扰动避免粒子陷入局部最优。最后强调了GSA在解决多峰优化问题方面的优势。 适合人群:对优化算法感兴趣的科研人员、学生及工程师,尤其是那些希望深入了解群体智能算法的人。 使用场景及目标:适用于需要高效寻找全局最优解的问题,特别是在面对复杂的多峰函数时。目标是帮助读者理解GSA的工作机制,掌握其MATLAB实现方法,并能够根据实际情况调整参数以获得更好的优化效果。 其他说明:尽管GSA在低维问题上有出色表现,但在高维优化问题中可能存在效率瓶颈,因此建议进一步研究并行计算或近似邻居搜索等改进措施。
基于Andorid的跨屏拖动应用设计实现源码,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。
DeepSeek R1 7b本地部署模型整合包及超全学习教程,资源总大小420G,喜欢的自行下载。
精品推荐,最新人工智能训练师认证资料汇总,15份。供大家学习参考。 (新版)人工智能训练师(中级)职业技能等级认定考试题库.pdf 2025年人工智能训练师(高级)职业技能鉴定参考题库(含答案).pdf 阿里认证高级人工智能训练师真题.pdf 初级人工智能训练师题库.pdf 高级人工智能化训练师认证答案解析.doc 高级人工智能训练师.docx 高级人工智能训练师题库.pdf 人工智能技术应用基础课件:人工智能训练师.pdf 人工智能训练师(服务机器人人工智能技术应用)(学生组)理论题库.pdf 人工智能训练师(服务机器人人工智能技术应用)理论题库.docx 人工智能训练师概述课件.pdf 人工智能训练师基础(上册).pdf 人工智能训练师技能等级认定四级理论知识试卷.docx 人工智能训练师试题及答案(150题).pdf 人工智能训练师职业技能标准.pdf
内容概要:本文探讨了Logistic函数在电力系统优化调度中的应用,特别是用于描述用户对电价变化的响应行为。文中详细介绍了Logistic函数如何通过S型曲线特性,将电价差与负荷转移率关联起来,形成死区、响应区和饱和区三个不同的响应阶段。此外,文章还展示了如何使用MATLAB进行仿真,以及在综合能源系统和微电网中的具体应用案例,如优化分时电价策略、设计需求响应激励机制等。 适合人群:电力系统研究人员、微电网调度工程师、能源管理专业学生。 使用场景及目标:适用于需要理解和应用需求响应模型的研究和工程项目,旨在提高电力系统的经济性和效率,优化调度策略。 其他说明:文章强调了模型的实际应用挑战,如参数调校、异常处理等,并提供了具体的MATLAB代码示例,帮助读者更好地理解和应用Logistic函数模型。
内容概要:本文档是一份C语言考核测试题,分为选择题和程序设计题两大部分。选择题部分共25题,涵盖C语言的基本概念、语法细节、运算符优先级、表达式求值、数据类型转换、控制结构等方面的知识点,旨在考察学生对C语言基础知识的理解与掌握。程序设计题部分提供了多个编程题目,如求数列和、阶乘之和、货币组合方式、质数与完数的求解、日期计算等,侧重于考察学生的实际编程能力和解决问题的能力。 适合人群:适合正在学习或复习C语言的学生,特别是计算机相关专业的本科生或高职高专学生。 使用场景及目标:①作为课堂练习或课后作业,帮助学生巩固所学知识;②作为考试或竞赛的模拟试题,评估学生对C语言的理解程度;③为教师提供教学参考,辅助课程设计与教学计划制定。 其他说明:建议考生在答题过程中仔细阅读题目要求,确保理解每个问题的具体含义。对于程序设计题,应先思考解决方案再动手编写代码,注意代码的规范性和可读性。同时,可以通过实际编译运行来验证程序的正确性。