package com.melnykov.fab;
import android.annotation.SuppressLint;
import android.content.Context;
import android.content.res.ColorStateList;
import android.content.res.TypedArray;
import android.graphics.Outline;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.LayerDrawable;
import android.graphics.drawable.RippleDrawable;
import android.graphics.drawable.ShapeDrawable;
import android.graphics.drawable.StateListDrawable;
import android.graphics.drawable.shapes.OvalShape;
import android.os.Build;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewOutlineProvider;
import android.view.ViewTreeObserver;
import android.view.animation.AnticipateInterpolator;
import android.view.animation.Interpolator;
import android.view.animation.OvershootInterpolator;
import android.widget.AbsListView;
import android.widget.ImageButton;
/**
* Android Google+ like floating action button which reacts on the attached list view scrolling events.
*
* @author Oleksandr Melnykov
*/
public class FloatingActionButton extends ImageButton {
private static final int TRANSLATE_DURATION_MILLIS = 200;
public static final int TYPE_NORMAL = 0;
public static final int TYPE_MINI = 1;
private boolean mVisible;
private int mColorNormal;
private int mColorPressed;
private int mColorRipple;
private boolean mShadow;
private int mType;
private int mShadowSize;
private int mScrollThreshold;
private boolean mMarginsSet;
private final Interpolator mShowInterpolator = new AnticipateInterpolator ();
private final Interpolator mHideInterpolator = new OvershootInterpolator ();
public FloatingActionButton(Context context) {
this(context, null);
}
public FloatingActionButton(Context context, AttributeSet attrs) {
super(context, attrs);
init(context, attrs);
}
public FloatingActionButton(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init(context, attrs);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int size = getDimension(
mType == TYPE_NORMAL ? R.dimen.fab_size_normal : R.dimen.fab_size_mini);
if (mShadow && !hasLollipopApi()) {
size += mShadowSize * 2;
setMarginsWithoutShadow();
}
setMeasuredDimension(size, size);
}
private void init(Context context, AttributeSet attributeSet) {
mVisible = true;
mColorNormal = getColor(R.color.material_blue_500);
mColorPressed = getColor(R.color.material_blue_600);
mColorRipple = getColor(android.R.color.white);
mType = TYPE_NORMAL;
mShadow = true;
mScrollThreshold = getResources().getDimensionPixelOffset(R.dimen.fab_scroll_threshold);
mShadowSize = getDimension(R.dimen.fab_shadow_size);
if (attributeSet != null) {
initAttributes(context, attributeSet);
}
updateBackground();
}
private void initAttributes(Context context, AttributeSet attributeSet) {
TypedArray attr = getTypedArray(context, attributeSet, R.styleable.FloatingActionButton);
if (attr != null) {
try {
mColorNormal = attr.getColor(R.styleable.FloatingActionButton_fab_colorNormal,
getColor(R.color.material_blue_500));
mColorPressed = attr.getColor(R.styleable.FloatingActionButton_fab_colorPressed,
getColor(R.color.material_blue_600));
mColorRipple = attr.getColor(R.styleable.FloatingActionButton_fab_colorRipple,
getColor(android.R.color.white));
mShadow = attr.getBoolean(R.styleable.FloatingActionButton_fab_shadow, true);
mType = attr.getInt(R.styleable.FloatingActionButton_fab_type, TYPE_NORMAL);
} finally {
attr.recycle();
}
}
}
private void updateBackground() {
StateListDrawable drawable = new StateListDrawable();
drawable.addState(new int[]{android.R.attr.state_pressed}, createDrawable(mColorPressed));
drawable.addState(new int[]{}, createDrawable(mColorNormal));
setBackgroundCompat(drawable);
}
private Drawable createDrawable(int color) {
OvalShape ovalShape = new OvalShape();
ShapeDrawable shapeDrawable = new ShapeDrawable(ovalShape);
shapeDrawable.getPaint().setColor(color);
if (mShadow && !hasLollipopApi()) {
Drawable shadowDrawable = getResources().getDrawable(mType == TYPE_NORMAL ? R.drawable.shadow
: R.drawable.shadow_mini);
LayerDrawable layerDrawable = new LayerDrawable(new Drawable[]{shadowDrawable, shapeDrawable});
layerDrawable.setLayerInset(1, mShadowSize, mShadowSize, mShadowSize, mShadowSize);
return layerDrawable;
} else {
return shapeDrawable;
}
}
private TypedArray getTypedArray(Context context, AttributeSet attributeSet, int[] attr) {
return context.obtainStyledAttributes(attributeSet, attr, 0, 0);
}
private int getColor(int id) {
return getResources().getColor(id);
}
private int getDimension(int id) {
return getResources().getDimensionPixelSize(id);
}
private void setMarginsWithoutShadow() {
if (!mMarginsSet) {
if (getLayoutParams() instanceof ViewGroup.MarginLayoutParams) {
ViewGroup.MarginLayoutParams layoutParams = (ViewGroup.MarginLayoutParams) getLayoutParams();
int leftMargin = layoutParams.leftMargin - mShadowSize;
int topMargin = layoutParams.topMargin - mShadowSize;
int rightMargin = layoutParams.rightMargin - mShadowSize;
int bottomMargin = layoutParams.bottomMargin - mShadowSize;
layoutParams.setMargins(leftMargin, topMargin, rightMargin, bottomMargin);
requestLayout();
mMarginsSet = true;
}
}
}
@SuppressWarnings("deprecation")
@SuppressLint("NewApi")
private void setBackgroundCompat(Drawable drawable) {
if (hasLollipopApi()) {
float elevation;
if (mShadow) {
elevation = getElevation() > 0.0f ? getElevation()
: getDimension(R.dimen.fab_elevation_lollipop);
} else {
elevation = 0.0f;
}
setElevation(elevation);
RippleDrawable rippleDrawable = new RippleDrawable(new ColorStateList(new int[][]{{}},
new int[]{mColorRipple}), drawable, null);
setOutlineProvider(new ViewOutlineProvider() {
@Override
public void getOutline(View view, Outline outline) {
int size = getDimension(mType == TYPE_NORMAL ? R.dimen.fab_size_normal : R.dimen.fab_size_mini);
outline.setOval(0, 0, size, size);
}
});
setClipToOutline(true);
setBackground(rippleDrawable);
} else if (hasJellyBeanApi()) {
setBackground(drawable);
} else {
setBackgroundDrawable(drawable);
}
}
private int getMarginBottom() {
int marginBottom = 0;
final ViewGroup.LayoutParams layoutParams = getLayoutParams();
if (layoutParams instanceof ViewGroup.MarginLayoutParams) {
marginBottom = ((ViewGroup.MarginLayoutParams) layoutParams).bottomMargin;
}
return marginBottom;
}
public void setColorNormal(int color) {
if (color != mColorNormal) {
mColorNormal = color;
updateBackground();
}
}
public void setColorNormalResId(int colorResId) {
setColorNormal(getColor(colorResId));
}
public int getColorNormal() {
return mColorNormal;
}
public void setColorPressed(int color) {
if (color != mColorPressed) {
mColorPressed = color;
updateBackground();
}
}
public void setColorPressedResId(int colorResId) {
setColorPressed(getColor(colorResId));
}
public int getColorPressed() {
return mColorPressed;
}
public void setColorRipple(int color) {
if (color != mColorRipple) {
mColorRipple = color;
updateBackground();
}
}
public void setColorRippleResId(int colorResId) {
setColorRipple(getColor(colorResId));
}
public int getColorRipple() {
return mColorRipple;
}
public void setShadow(boolean shadow) {
if (shadow != mShadow) {
mShadow = shadow;
updateBackground();
}
}
public boolean hasShadow() {
return mShadow;
}
public void setType(int type) {
if (type != mType) {
mType = type;
updateBackground();
}
}
public int getType() {
return mType;
}
public boolean isVisible() {
return mVisible;
}
public void show() {
show(true);
}
public void hide() {
hide(true);
}
public void show(boolean animate) {
toggle(true, animate, false);
}
public void hide(boolean animate) {
toggle(false, animate, false);
}
private void toggle(final boolean visible, final boolean animate, boolean force) {
if (mVisible != visible || force) {
mVisible = visible;
int height = getHeight();
if (height == 0 && !force) {
ViewTreeObserver vto = getViewTreeObserver();
if (vto.isAlive()) {
vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
@Override
public boolean onPreDraw() {
ViewTreeObserver currentVto = getViewTreeObserver();
if (currentVto.isAlive()) {
currentVto.removeOnPreDrawListener(this);
}
toggle(visible, animate, true);
return true;
}
});
return;
}
}
//int translationY = visible ? 0 : height + getMarginBottom();
long scale = visible?1:0;
if (animate) {
this.animate().setInterpolator(visible?mHideInterpolator:mShowInterpolator)
.setDuration(TRANSLATE_DURATION_MILLIS)
.scaleX(scale).scaleY(scale);
//.translationY(translationY);
} else {
//this.setTranslationY(translationY);
this.setScaleX(scale);
this.setScaleY(scale);
}
// On pre-Honeycomb a translated view is still clickable, so we need to disable clicks manually
// if (!hasHoneycombApi()) {
setClickable(visible);
// }
}
}
public AbsListView.OnScrollListener attachToListView(AbsListView listView) {
return attachToListView(listView, null);
}
public ObservableScrollView.OnScrollChangedListener attachToScrollView(ObservableScrollView scrollView) {
return attachToScrollView(scrollView, null);
}
public AbsListView.OnScrollListener attachToListView(AbsListView listView, ScrollDirectionListener listener) {
return attachToListView(listView,listener,true);
}
public AbsListView.OnScrollListener attachToListView(AbsListView listView, ScrollDirectionListener listener,boolean insert) {
AbsListViewScrollDetectorImpl scrollDetector = new AbsListViewScrollDetectorImpl();
scrollDetector.setListener(listener);
scrollDetector.setListView(listView);
scrollDetector.setScrollThreshold(mScrollThreshold);
if(insert) {
listView.setOnScrollListener(scrollDetector);
}
return scrollDetector;
}
public ObservableScrollView.OnScrollChangedListener attachToScrollView( ObservableScrollView scrollView, ScrollDirectionListener listener) {
return attachToScrollView(scrollView, listener,true);
}
public ObservableScrollView.OnScrollChangedListener attachToScrollView( ObservableScrollView scrollView, ScrollDirectionListener listener,boolean insert) {
ScrollViewScrollDetectorImpl scrollDetector = new ScrollViewScrollDetectorImpl();
scrollDetector.setListener(listener);
scrollDetector.setScrollThreshold(mScrollThreshold);
if (insert) {
scrollView.setOnScrollChangedListener(scrollDetector);
}
return scrollDetector;
}
private boolean hasLollipopApi() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP;
}
private boolean hasJellyBeanApi() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN;
}
private boolean hasHoneycombApi() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB;
}
private class AbsListViewScrollDetectorImpl extends AbsListViewScrollDetector {
private ScrollDirectionListener mListener;
private void setListener(ScrollDirectionListener scrollDirectionListener) {
mListener = scrollDirectionListener;
}
@Override
public void onScrollDown() {
show();
if (mListener != null) {
mListener.onScrollDown();
}
}
@Override
void onScrollIdle() {
//show();
}
@Override
public void onScrollUp() {
hide();
if (mListener != null) {
mListener.onScrollUp();
}
}
}
private class ScrollViewScrollDetectorImpl extends ScrollViewScrollDetector {
private ScrollDirectionListener mListener;
private void setListener(ScrollDirectionListener scrollDirectionListener) {
mListener = scrollDirectionListener;
}
@Override
public void onScrollDown() {
show();
if (mListener != null) {
mListener.onScrollDown();
}
}
@Override
public void onScrollUp() {
hide();
if (mListener != null) {
mListener.onScrollUp();
}
}
}
}