package com.example.widget;
import android.animation.Animator;
import android.content.Context;
import android.support.design.widget.CoordinatorLayout;
import android.support.v4.view.ViewCompat;
import android.support.v4.view.animation.FastOutSlowInInterpolator;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewPropertyAnimator;
import android.view.animation.Interpolator;
/**
* CoordinatorLayout Behavior for a quick return footer
* <p>
* When a nested ScrollView is scrolled down, the quick return view will disappear.
* When the ScrollView is scrolled back up, the quick return view will reappear.
* <p>
* Source: https://github.com/bherbst/QuickReturnFooterSample
*
* @author bherbst
*/
@SuppressWarnings("unused")
public class QuickReturnFooterBehavior extends CoordinatorLayout.Behavior<View>
{
private static final Interpolator INTERPOLATOR = new FastOutSlowInInterpolator();
private int mDySinceDirectionChange;
private boolean mIsShowing;
private boolean mIsHiding;
public QuickReturnFooterBehavior(Context context, AttributeSet attrs)
{
super(context, attrs);
}
@Override
public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, View child, View directTargetChild, View target, int nestedScrollAxes)
{
return (nestedScrollAxes & ViewCompat.SCROLL_AXIS_VERTICAL) != 0;
}
@Override
public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, View child, View target, int dx, int dy, int[] consumed)
{
if(dy > 0 && mDySinceDirectionChange < 0 || dy < 0 && mDySinceDirectionChange > 0)
{
// We detected a direction change- cancel existing animations and reset our cumulative delta Y
child.animate().cancel();
mDySinceDirectionChange = 0;
}
mDySinceDirectionChange += dy;
if(mDySinceDirectionChange > child.getHeight() && child.getVisibility() == View.VISIBLE && !mIsHiding)
{
hide(child);
}
else if(mDySinceDirectionChange < 0 && child.getVisibility() == View.GONE && !mIsShowing)
{
show(child);
}
}
/**
* Hide the quick return view.
* <p>
* Animates hiding the view, with the view sliding down and out of the screen.
* After the view has disappeared, its visibility will change to GONE.
*
* @param view The quick return view
*/
private void hide(final View view)
{
mIsHiding = true;
ViewPropertyAnimator animator = view.animate()
.translationY(view.getHeight())
.setInterpolator(INTERPOLATOR)
.setDuration(200);
animator.setListener(new Animator.AnimatorListener()
{
@Override
public void onAnimationStart(Animator animator) {}
@Override
public void onAnimationEnd(Animator animator)
{
// Prevent drawing the View after it is gone
mIsHiding = false;
view.setVisibility(View.GONE);
}
@Override
public void onAnimationCancel(Animator animator)
{
// Canceling a hide should show the view
mIsHiding = false;
if(!mIsShowing)
{
show(view);
}
}
@Override
public void onAnimationRepeat(Animator animator) {}
});
animator.start();
}
/**
* Show the quick return view.
* <p>
* Animates showing the view, with the view sliding up from the bottom of the screen.
* After the view has reappeared, its visibility will change to VISIBLE.
*
* @param view The quick return view
*/
private void show(final View view)
{
mIsShowing = true;
ViewPropertyAnimator animator = view.animate()
.translationY(0)
.setInterpolator(INTERPOLATOR)
.setDuration(200);
animator.setListener(new Animator.AnimatorListener()
{
@Override
public void onAnimationStart(Animator animator)
{
view.setVisibility(View.VISIBLE);
}
@Override
public void onAnimationEnd(Animator animator)
{
mIsShowing = false;
}
@Override
public void onAnimationCancel(Animator animator)
{
// Canceling a show should hide the view
mIsShowing = false;
if(!mIsHiding)
{
hide(view);
}
}
@Override
public void onAnimationRepeat(Animator animator) {}
});
animator.start();
}
}