Хореография - Push-анимация для Recycler View
Я пытаюсь динамически вставлять элементы в представление переработчика, основываясь на действиях пользователя. И при вставке или удалении элемента я намереваюсь использовать анимацию хореографии, где входящий элемент перемещает другие элементы вверх и вниз. Анимация, которую я собираюсь сделать, доступна по ссылке
Будет очень полезно, если мне предоставят информацию о том, как продвигаться с этим типом анимации.
1 ответ
Попробуй это:
public class AdapterRow extends RecyclerView.Adapter<AdapterPosliteNamFoto.ItemHolder> {
private final ArrayList<DataHolder> array;
public AdapterRow() {
this.array = new ArrayList<>();
this.array.add(new DataHolder("Text 1"));
this.array.add(new DataHolder("Text 2"));
this.array.add(new DataHolder("Text 3"));
}
@Override
public int getItemCount() {
return (null != array ? array.size() : 0);
}
@NonNull
@Override
public ItemHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
return new ItemHolder(LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.item_row, viewGroup, false));
}
@Override
public void onBindViewHolder(@NonNull final ItemHolder holder, int position) {
DataHolder model = array.get(position);
holder.txt.setText(model.txt);
holder.rootLt.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
int pos = holder.getAdapterPosition();
if (pos != RecyclerView.NO_POSITION) {
DataHolder model = array.get(pos);
if (model.rowState == RowState.EXPANDED) {
changeRowState(holder, pos, model, RowState.COLLAPSING);
} else if (model.rowState == RowState.COLLAPSED) {
changeRowState(holder, pos, model, RowState.EXPANDING);
}
}
}
});
}
public void changeRowState(ItemHolder holder, int pos, DataHolder model, @RowState int rowState) {
model.rowState = rowState;
array.set(pos, model);
final View v = holder.rootLt;
if (rowState == RowState.EXPANDING) {
v.measure(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
final int targetHeight = v.getMeasuredHeight() * 2;
v.getLayoutParams().height = 1;
v.setVisibility(View.VISIBLE);
ValueAnimator va = ValueAnimator.ofInt(1, targetHeight);
va.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
public void onAnimationUpdate(ValueAnimator animation) {
v.getLayoutParams().height = (Integer) animation.getAnimatedValue();
v.requestLayout();
}
});
va.addListener(new Animator.AnimatorListener() {
@Override
public void onAnimationEnd(Animator animation) {
v.getLayoutParams().height = ViewGroup.LayoutParams.WRAP_CONTENT;
}
@Override public void onAnimationStart(Animator animation) {}
@Override public void onAnimationCancel(Animator animation) {}
@Override public void onAnimationRepeat(Animator animation) {}
});
va.setDuration(300);
va.setInterpolator(new OvershootInterpolator());
va.start();
} else if (rowState == RowState.COLLAPSING) {
final int initialHeight = v.getMeasuredHeight();
ValueAnimator va = ValueAnimator.ofInt(initialHeight, initialHeight / 2);
va.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
public void onAnimationUpdate(ValueAnimator animation) {
v.getLayoutParams().height = (Integer) animation.getAnimatedValue();
v.requestLayout();
}
});
va.setDuration(300);
va.setInterpolator(new DecelerateInterpolator());
va.start();
}
}
public class ItemHolder extends RecyclerView.ViewHolder {
protected ViewGroup rootLt;
protected TextView txt;
public ItemHolder(View view) {
super(view);
this.rootLt = view.findViewById(R.id.root_layout);
this.txt = view.findViewById(R.id.txt);
}
}
@IntDef(value = {
RowState.COLLAPSED, RowState.COLLAPSING, RowState.EXPANDED, RowState.EXPANDING})
@Retention(RetentionPolicy.SOURCE)
public @interface RowState {
int COLLAPSED = 0;
int COLLAPSING = 1;
int EXPANDED = 2;
int EXPANDING = 3;
}
public class DataHolder {
public String text;
@RowState
public int rowState = RowState.Collapsed;
public DataHolder(String text) {
this.text = text;
}
}
}