android View 关于transient
今天来研究一下 ListView 的删除动画
由于 ListView 卷动时会把画面上的 item 重用以显示不同数据
这样会导致我们可能会删除到非正确的 item
或是出现显示上的问题(该 item 显示的数据已经不同但是动画却还在它上面跑)
要解决这个问题就要使用 View 类别在 API 16 加入的新 method
public void setHasTransientState (boolean hasTransientState)
当设为 true 的时候,就告诉系统这个 View 应该尽可能的被保留,直到setHasTransientState(false)被呼叫
要注意这是可以重复设置的,也就是说每一个setHasTransientState(true)都要搭配一个setHasTransientState(false)才能回复
//Listview 的 OnItemClickListener 的内容 //本范例点了 item 后会淡出并删除该 item public void onItemClick(AdapterView parent, final View view, int position, long id) { final String item = (String) parent.getItemAtPosition(position); ObjectAnimator anim = ObjectAnimator.ofFloat(view, View.ALPHA, 0); anim.setDuration(500); view.setHasTransientState(true); //设为 true 宣告 item 要被追踪 anim.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { myListview.remove(item); adapter.notifyDataSetChanged(); //重新整理 listview view.setAlpha(1); view.setHasTransientState(false); //完成后设定回 false } }); anim.start(); }
另一种方法是用 ViewPropertyAnimator 来解决
public void onItemClick(AdapterView parent, final View view, int position, long id) { final String item = (String) parent.getItemAtPosition(position); view.animate() .setDuration(500) .alpha(0) .withEndAction(new Runnable() { @Override public void run() { myListview.remove(item); adapter.notifyDataSetChanged(); view.setAlpha(1); } } ); }
但以上两种方式最大的问题就是 API 16 以后才能使用(用到了 withEndAction 跟 setHasTransientState)
(该死的 Android Fragmentation…)
让我们看看另一个例子
在 DevBytes: Animating ListView Deletion: Now on Gingerbread! 里
他用了一个继承自 ArrayAdapter 的自定义 adapter
override 掉 public boolean hasStableIds() 让它总是回传 true ,用来取代 setHasTransientState(true)
//用来取代 setHasTransientState(true) @Override public boolean hasStableIds() { return true; //总是回传 true,用来取代 setHasTransientState(true) }
而 withEndAction 的部分则用 setAnimationListener 取代
//API 11 之前的方法 TranslateAnimation translator = new TranslateAnimation(startX, endX, startY, endY); translator.setDuration(150); view.startAnimation(translator); view.getAnimation().setAnimationListener(new AnimationListenerAdapter() { @Override public void onAnimationEnd(Animation animation) { new Runnable() { @Override public void run() { //...(略) } } } });
这样可以让 Gingerbread(API 10) 或更早的 Android 版本兼容
如果是 4.0 (API 14)之后的版本
还是用 view.animate() 跟 ObjectAnimator 来做
//API 小于 16 但大于等于 14 可以这样做 view.animate().setDuration(150); ObjectAnimator anim = ObjectAnimator.ofFloat(view, View.TRANSLATION_X, startX, endX); anim.setDuration(150); anim.start(); setAnimatorEndAction(anim, new Runnable() { @Override public void run() { //...(略) } });