RecyclerViewで区切り線を引いてアニメーションさせる
RecyclerViewで区切り線
ListViewでデフォルトで表示されるあれです。
RecyclerViewだと、デフォルトでは表示されないので、表示されるように実装が必要です。RecyclerViewのバージョンは23.1.0
です。
RecyclerViewで行間に区切り線を表示するにあるようにすることで、区切り線を表示できました。
追加・削除アニメーション
RecyclerViewだと、アニメーションが簡単に実装できます。しかし、上記のやり方だと、アニメーション時に区切り線が動かず、セルの中身のみが動く挙動になりました。Googleのサンプルにあるクラスを用いることで、治りました。
DividerItemDecoration.java
古いサンプルクラスを用いると、動かないのにハマりました。 同じようにハマっている人がいたので、クラスのどこが変わったのか差分をとりました。
$ git diff 18728e9dd5dd66d4f5edf1b792e77e2b544a1cb0..882ee91503f28d1fd60ba939c5b37db6b9d4ad29 DividerItemDecoration.java diff --git a/sdk/extras/android/support/samples/Support7Demos/src/com/example/android/supportv7/widget/decorator/DividerItemDecoration.java b/sdk/extras/android/support/samples/Support7Demos/src/com/example/android/supportv7/widget/decorator/DividerItemDecoration.java index 4d5d208..4386f4f 100644 --- a/sdk/extras/android/support/samples/Support7Demos/src/com/example/android/supportv7/widget/decorator/DividerItemDecoration.java +++ b/sdk/extras/android/support/samples/Support7Demos/src/com/example/android/supportv7/widget/decorator/DividerItemDecoration.java @@ -21,6 +21,7 @@ import android.content.res.TypedArray; import android.graphics.Canvas; import android.graphics.Rect; import android.graphics.drawable.Drawable; +import android.support.v4.view.ViewCompat; import android.support.v7.widget.LinearLayoutManager; import android.support.v7.widget.RecyclerView; import android.view.View; @@ -71,7 +72,8 @@ public class DividerItemDecoration extends RecyclerView.ItemDecoration { final View child = parent.getChildAt(i); final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child .getLayoutParams(); - final int top = child.getBottom() + params.bottomMargin; + final int top = child.getBottom() + params.bottomMargin + + Math.round(ViewCompat.getTranslationY(child)); final int bottom = top + mDivider.getIntrinsicHeight(); mDivider.setBounds(left, top, right, bottom); mDivider.draw(c); @@ -87,7 +89,8 @@ public class DividerItemDecoration extends RecyclerView.ItemDecoration { final View child = parent.getChildAt(i); final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child .getLayoutParams(); - final int left = child.getRight() + params.rightMargin; + final int left = child.getRight() + params.rightMargin + + Math.round(ViewCompat.getTranslationX(child)); final int right = left + mDivider.getIntrinsicHeight(); mDivider.setBounds(left, top, right, bottom); mDivider.draw(c); (END)
縦横それぞれで、描画時に、追加・削除で変化があった分を考慮されています。
onDraw(Canvas c, RecyclerView parent)
とgetItemOffsets(Rect outRect, int itemPosition, RecyclerView parent)
が、Deprecatedになっていたので、その部分を修正したものが次のコードです。
public class DividerItemDecoration extends RecyclerView.ItemDecoration { private static final int[] ATTRS = new int[]{ android.R.attr.listDivider }; public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL; public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL; private Drawable mDivider; private int mOrientation; public DividerItemDecoration(Context context, int orientation) { final TypedArray a = context.obtainStyledAttributes(ATTRS); mDivider = a.getDrawable(0); a.recycle(); setOrientation(orientation); } public void setOrientation(int orientation) { if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) { throw new IllegalArgumentException("invalid orientation"); } mOrientation = orientation; } @Override public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) { if (mOrientation == VERTICAL_LIST) { drawVertical(c, parent); } else { drawHorizontal(c, parent); } } public void drawVertical(Canvas c, RecyclerView parent) { final int left = parent.getPaddingLeft(); final int right = parent.getWidth() - parent.getPaddingRight(); final int childCount = parent.getChildCount(); for (int i = 0; i < childCount; i++) { final View child = parent.getChildAt(i); final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child .getLayoutParams(); final int top = child.getBottom() + params.bottomMargin + Math.round(ViewCompat.getTranslationY(child)); final int bottom = top + mDivider.getIntrinsicHeight(); mDivider.setBounds(left, top, right, bottom); mDivider.draw(c); } } public void drawHorizontal(Canvas c, RecyclerView parent) { final int top = parent.getPaddingTop(); final int bottom = parent.getHeight() - parent.getPaddingBottom(); final int childCount = parent.getChildCount(); for (int i = 0; i < childCount; i++) { final View child = parent.getChildAt(i); final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child .getLayoutParams(); final int left = child.getRight() + params.rightMargin + Math.round(ViewCompat.getTranslationX(child)); final int right = left + mDivider.getIntrinsicHeight(); mDivider.setBounds(left, top, right, bottom); mDivider.draw(c); } } @Override public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { if (mOrientation == VERTICAL_LIST) { outRect.set(0, 0, 0, mDivider.getIntrinsicHeight()); } else { outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0); } } }
本当に差し替えただけですが、以上です。
参考
RecyclerViewで行間に区切り線を表示する / Qiita
chromium/android_tools / Google Git
特定ファイルの前回(直前)の変更とのdiffをみる / Qiita
RecyclerView: ItemDecorations aren't animated with ItemAnimators / reddit