| /* |
| * Copyright (C) 2015 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package android.support.design.widget; |
| |
| import android.animation.Animator; |
| import android.animation.AnimatorListenerAdapter; |
| import android.animation.ValueAnimator; |
| import android.content.Context; |
| import android.content.res.ColorStateList; |
| import android.graphics.Color; |
| import android.graphics.PorterDuff; |
| import android.graphics.Rect; |
| import android.graphics.drawable.Drawable; |
| import android.graphics.drawable.GradientDrawable; |
| import android.graphics.drawable.LayerDrawable; |
| import android.os.Build; |
| import android.support.annotation.NonNull; |
| import android.support.annotation.Nullable; |
| import android.support.annotation.RequiresApi; |
| import android.support.design.R; |
| import android.support.v4.content.ContextCompat; |
| import android.support.v4.graphics.drawable.DrawableCompat; |
| import android.support.v4.view.ViewCompat; |
| import android.view.View; |
| import android.view.ViewTreeObserver; |
| import android.view.animation.Interpolator; |
| |
| @RequiresApi(14) |
| class FloatingActionButtonImpl { |
| static final Interpolator ANIM_INTERPOLATOR = AnimationUtils.FAST_OUT_LINEAR_IN_INTERPOLATOR; |
| static final long PRESSED_ANIM_DURATION = 100; |
| static final long PRESSED_ANIM_DELAY = 100; |
| |
| static final int ANIM_STATE_NONE = 0; |
| static final int ANIM_STATE_HIDING = 1; |
| static final int ANIM_STATE_SHOWING = 2; |
| |
| int mAnimState = ANIM_STATE_NONE; |
| |
| private final StateListAnimator mStateListAnimator; |
| |
| ShadowDrawableWrapper mShadowDrawable; |
| |
| private float mRotation; |
| |
| Drawable mShapeDrawable; |
| Drawable mRippleDrawable; |
| CircularBorderDrawable mBorderDrawable; |
| Drawable mContentBackground; |
| |
| float mElevation; |
| float mPressedTranslationZ; |
| |
| interface InternalVisibilityChangedListener { |
| void onShown(); |
| void onHidden(); |
| } |
| |
| static final int SHOW_HIDE_ANIM_DURATION = 200; |
| |
| static final int[] PRESSED_ENABLED_STATE_SET = {android.R.attr.state_pressed, |
| android.R.attr.state_enabled}; |
| static final int[] FOCUSED_ENABLED_STATE_SET = {android.R.attr.state_focused, |
| android.R.attr.state_enabled}; |
| static final int[] ENABLED_STATE_SET = {android.R.attr.state_enabled}; |
| static final int[] EMPTY_STATE_SET = new int[0]; |
| |
| final VisibilityAwareImageButton mView; |
| final ShadowViewDelegate mShadowViewDelegate; |
| |
| private final Rect mTmpRect = new Rect(); |
| private ViewTreeObserver.OnPreDrawListener mPreDrawListener; |
| |
| FloatingActionButtonImpl(VisibilityAwareImageButton view, |
| ShadowViewDelegate shadowViewDelegate) { |
| mView = view; |
| mShadowViewDelegate = shadowViewDelegate; |
| |
| mStateListAnimator = new StateListAnimator(); |
| |
| // Elevate with translationZ when pressed or focused |
| mStateListAnimator.addState(PRESSED_ENABLED_STATE_SET, |
| createAnimator(new ElevateToTranslationZAnimation())); |
| mStateListAnimator.addState(FOCUSED_ENABLED_STATE_SET, |
| createAnimator(new ElevateToTranslationZAnimation())); |
| // Reset back to elevation by default |
| mStateListAnimator.addState(ENABLED_STATE_SET, |
| createAnimator(new ResetElevationAnimation())); |
| // Set to 0 when disabled |
| mStateListAnimator.addState(EMPTY_STATE_SET, |
| createAnimator(new DisabledElevationAnimation())); |
| |
| mRotation = mView.getRotation(); |
| } |
| |
| void setBackgroundDrawable(ColorStateList backgroundTint, |
| PorterDuff.Mode backgroundTintMode, int rippleColor, int borderWidth) { |
| // Now we need to tint the original background with the tint, using |
| // an InsetDrawable if we have a border width |
| mShapeDrawable = DrawableCompat.wrap(createShapeDrawable()); |
| DrawableCompat.setTintList(mShapeDrawable, backgroundTint); |
| if (backgroundTintMode != null) { |
| DrawableCompat.setTintMode(mShapeDrawable, backgroundTintMode); |
| } |
| |
| // Now we created a mask Drawable which will be used for touch feedback. |
| GradientDrawable touchFeedbackShape = createShapeDrawable(); |
| |
| // We'll now wrap that touch feedback mask drawable with a ColorStateList. We do not need |
| // to inset for any border here as LayerDrawable will nest the padding for us |
| mRippleDrawable = DrawableCompat.wrap(touchFeedbackShape); |
| DrawableCompat.setTintList(mRippleDrawable, createColorStateList(rippleColor)); |
| |
| final Drawable[] layers; |
| if (borderWidth > 0) { |
| mBorderDrawable = createBorderDrawable(borderWidth, backgroundTint); |
| layers = new Drawable[] {mBorderDrawable, mShapeDrawable, mRippleDrawable}; |
| } else { |
| mBorderDrawable = null; |
| layers = new Drawable[] {mShapeDrawable, mRippleDrawable}; |
| } |
| |
| mContentBackground = new LayerDrawable(layers); |
| |
| mShadowDrawable = new ShadowDrawableWrapper( |
| mView.getContext(), |
| mContentBackground, |
| mShadowViewDelegate.getRadius(), |
| mElevation, |
| mElevation + mPressedTranslationZ); |
| mShadowDrawable.setAddPaddingForCorners(false); |
| mShadowViewDelegate.setBackgroundDrawable(mShadowDrawable); |
| } |
| |
| void setBackgroundTintList(ColorStateList tint) { |
| if (mShapeDrawable != null) { |
| DrawableCompat.setTintList(mShapeDrawable, tint); |
| } |
| if (mBorderDrawable != null) { |
| mBorderDrawable.setBorderTint(tint); |
| } |
| } |
| |
| void setBackgroundTintMode(PorterDuff.Mode tintMode) { |
| if (mShapeDrawable != null) { |
| DrawableCompat.setTintMode(mShapeDrawable, tintMode); |
| } |
| } |
| |
| |
| void setRippleColor(int rippleColor) { |
| if (mRippleDrawable != null) { |
| DrawableCompat.setTintList(mRippleDrawable, createColorStateList(rippleColor)); |
| } |
| } |
| |
| final void setElevation(float elevation) { |
| if (mElevation != elevation) { |
| mElevation = elevation; |
| onElevationsChanged(elevation, mPressedTranslationZ); |
| } |
| } |
| |
| float getElevation() { |
| return mElevation; |
| } |
| |
| final void setPressedTranslationZ(float translationZ) { |
| if (mPressedTranslationZ != translationZ) { |
| mPressedTranslationZ = translationZ; |
| onElevationsChanged(mElevation, translationZ); |
| } |
| } |
| |
| void onElevationsChanged(float elevation, float pressedTranslationZ) { |
| if (mShadowDrawable != null) { |
| mShadowDrawable.setShadowSize(elevation, elevation + mPressedTranslationZ); |
| updatePadding(); |
| } |
| } |
| |
| void onDrawableStateChanged(int[] state) { |
| mStateListAnimator.setState(state); |
| } |
| |
| void jumpDrawableToCurrentState() { |
| mStateListAnimator.jumpToCurrentState(); |
| } |
| |
| void hide(@Nullable final InternalVisibilityChangedListener listener, final boolean fromUser) { |
| if (isOrWillBeHidden()) { |
| // We either are or will soon be hidden, skip the call |
| return; |
| } |
| |
| mView.animate().cancel(); |
| |
| if (shouldAnimateVisibilityChange()) { |
| mAnimState = ANIM_STATE_HIDING; |
| |
| mView.animate() |
| .scaleX(0f) |
| .scaleY(0f) |
| .alpha(0f) |
| .setDuration(SHOW_HIDE_ANIM_DURATION) |
| .setInterpolator(AnimationUtils.FAST_OUT_LINEAR_IN_INTERPOLATOR) |
| .setListener(new AnimatorListenerAdapter() { |
| private boolean mCancelled; |
| |
| @Override |
| public void onAnimationStart(Animator animation) { |
| mView.internalSetVisibility(View.VISIBLE, fromUser); |
| mCancelled = false; |
| } |
| |
| @Override |
| public void onAnimationCancel(Animator animation) { |
| mCancelled = true; |
| } |
| |
| @Override |
| public void onAnimationEnd(Animator animation) { |
| mAnimState = ANIM_STATE_NONE; |
| |
| if (!mCancelled) { |
| mView.internalSetVisibility(fromUser ? View.GONE : View.INVISIBLE, |
| fromUser); |
| if (listener != null) { |
| listener.onHidden(); |
| } |
| } |
| } |
| }); |
| } else { |
| // If the view isn't laid out, or we're in the editor, don't run the animation |
| mView.internalSetVisibility(fromUser ? View.GONE : View.INVISIBLE, fromUser); |
| if (listener != null) { |
| listener.onHidden(); |
| } |
| } |
| } |
| |
| void show(@Nullable final InternalVisibilityChangedListener listener, final boolean fromUser) { |
| if (isOrWillBeShown()) { |
| // We either are or will soon be visible, skip the call |
| return; |
| } |
| |
| mView.animate().cancel(); |
| |
| if (shouldAnimateVisibilityChange()) { |
| mAnimState = ANIM_STATE_SHOWING; |
| |
| if (mView.getVisibility() != View.VISIBLE) { |
| // If the view isn't visible currently, we'll animate it from a single pixel |
| mView.setAlpha(0f); |
| mView.setScaleY(0f); |
| mView.setScaleX(0f); |
| } |
| |
| mView.animate() |
| .scaleX(1f) |
| .scaleY(1f) |
| .alpha(1f) |
| .setDuration(SHOW_HIDE_ANIM_DURATION) |
| .setInterpolator(AnimationUtils.LINEAR_OUT_SLOW_IN_INTERPOLATOR) |
| .setListener(new AnimatorListenerAdapter() { |
| @Override |
| public void onAnimationStart(Animator animation) { |
| mView.internalSetVisibility(View.VISIBLE, fromUser); |
| } |
| |
| @Override |
| public void onAnimationEnd(Animator animation) { |
| mAnimState = ANIM_STATE_NONE; |
| if (listener != null) { |
| listener.onShown(); |
| } |
| } |
| }); |
| } else { |
| mView.internalSetVisibility(View.VISIBLE, fromUser); |
| mView.setAlpha(1f); |
| mView.setScaleY(1f); |
| mView.setScaleX(1f); |
| if (listener != null) { |
| listener.onShown(); |
| } |
| } |
| } |
| |
| final Drawable getContentBackground() { |
| return mContentBackground; |
| } |
| |
| void onCompatShadowChanged() { |
| // Ignore pre-v21 |
| } |
| |
| final void updatePadding() { |
| Rect rect = mTmpRect; |
| getPadding(rect); |
| onPaddingUpdated(rect); |
| mShadowViewDelegate.setShadowPadding(rect.left, rect.top, rect.right, rect.bottom); |
| } |
| |
| void getPadding(Rect rect) { |
| mShadowDrawable.getPadding(rect); |
| } |
| |
| void onPaddingUpdated(Rect padding) {} |
| |
| void onAttachedToWindow() { |
| if (requirePreDrawListener()) { |
| ensurePreDrawListener(); |
| mView.getViewTreeObserver().addOnPreDrawListener(mPreDrawListener); |
| } |
| } |
| |
| void onDetachedFromWindow() { |
| if (mPreDrawListener != null) { |
| mView.getViewTreeObserver().removeOnPreDrawListener(mPreDrawListener); |
| mPreDrawListener = null; |
| } |
| } |
| |
| boolean requirePreDrawListener() { |
| return true; |
| } |
| |
| CircularBorderDrawable createBorderDrawable(int borderWidth, ColorStateList backgroundTint) { |
| final Context context = mView.getContext(); |
| CircularBorderDrawable borderDrawable = newCircularDrawable(); |
| borderDrawable.setGradientColors( |
| ContextCompat.getColor(context, R.color.design_fab_stroke_top_outer_color), |
| ContextCompat.getColor(context, R.color.design_fab_stroke_top_inner_color), |
| ContextCompat.getColor(context, R.color.design_fab_stroke_end_inner_color), |
| ContextCompat.getColor(context, R.color.design_fab_stroke_end_outer_color)); |
| borderDrawable.setBorderWidth(borderWidth); |
| borderDrawable.setBorderTint(backgroundTint); |
| return borderDrawable; |
| } |
| |
| CircularBorderDrawable newCircularDrawable() { |
| return new CircularBorderDrawable(); |
| } |
| |
| void onPreDraw() { |
| final float rotation = mView.getRotation(); |
| if (mRotation != rotation) { |
| mRotation = rotation; |
| updateFromViewRotation(); |
| } |
| } |
| |
| private void ensurePreDrawListener() { |
| if (mPreDrawListener == null) { |
| mPreDrawListener = new ViewTreeObserver.OnPreDrawListener() { |
| @Override |
| public boolean onPreDraw() { |
| FloatingActionButtonImpl.this.onPreDraw(); |
| return true; |
| } |
| }; |
| } |
| } |
| |
| GradientDrawable createShapeDrawable() { |
| GradientDrawable d = newGradientDrawableForShape(); |
| d.setShape(GradientDrawable.OVAL); |
| d.setColor(Color.WHITE); |
| return d; |
| } |
| |
| GradientDrawable newGradientDrawableForShape() { |
| return new GradientDrawable(); |
| } |
| |
| boolean isOrWillBeShown() { |
| if (mView.getVisibility() != View.VISIBLE) { |
| // If we not currently visible, return true if we're animating to be shown |
| return mAnimState == ANIM_STATE_SHOWING; |
| } else { |
| // Otherwise if we're visible, return true if we're not animating to be hidden |
| return mAnimState != ANIM_STATE_HIDING; |
| } |
| } |
| |
| boolean isOrWillBeHidden() { |
| if (mView.getVisibility() == View.VISIBLE) { |
| // If we currently visible, return true if we're animating to be hidden |
| return mAnimState == ANIM_STATE_HIDING; |
| } else { |
| // Otherwise if we're not visible, return true if we're not animating to be shown |
| return mAnimState != ANIM_STATE_SHOWING; |
| } |
| } |
| |
| private ValueAnimator createAnimator(@NonNull ShadowAnimatorImpl impl) { |
| final ValueAnimator animator = new ValueAnimator(); |
| animator.setInterpolator(ANIM_INTERPOLATOR); |
| animator.setDuration(PRESSED_ANIM_DURATION); |
| animator.addListener(impl); |
| animator.addUpdateListener(impl); |
| animator.setFloatValues(0, 1); |
| return animator; |
| } |
| |
| private abstract class ShadowAnimatorImpl extends AnimatorListenerAdapter |
| implements ValueAnimator.AnimatorUpdateListener { |
| private boolean mValidValues; |
| private float mShadowSizeStart; |
| private float mShadowSizeEnd; |
| |
| @Override |
| public void onAnimationUpdate(ValueAnimator animator) { |
| if (!mValidValues) { |
| mShadowSizeStart = mShadowDrawable.getShadowSize(); |
| mShadowSizeEnd = getTargetShadowSize(); |
| mValidValues = true; |
| } |
| |
| mShadowDrawable.setShadowSize(mShadowSizeStart |
| + ((mShadowSizeEnd - mShadowSizeStart) * animator.getAnimatedFraction())); |
| } |
| |
| @Override |
| public void onAnimationEnd(Animator animator) { |
| mShadowDrawable.setShadowSize(mShadowSizeEnd); |
| mValidValues = false; |
| } |
| |
| /** |
| * @return the shadow size we want to animate to. |
| */ |
| protected abstract float getTargetShadowSize(); |
| } |
| |
| private class ResetElevationAnimation extends ShadowAnimatorImpl { |
| ResetElevationAnimation() { |
| } |
| |
| @Override |
| protected float getTargetShadowSize() { |
| return mElevation; |
| } |
| } |
| |
| private class ElevateToTranslationZAnimation extends ShadowAnimatorImpl { |
| ElevateToTranslationZAnimation() { |
| } |
| |
| @Override |
| protected float getTargetShadowSize() { |
| return mElevation + mPressedTranslationZ; |
| } |
| } |
| |
| private class DisabledElevationAnimation extends ShadowAnimatorImpl { |
| DisabledElevationAnimation() { |
| } |
| |
| @Override |
| protected float getTargetShadowSize() { |
| return 0f; |
| } |
| } |
| |
| private static ColorStateList createColorStateList(int selectedColor) { |
| final int[][] states = new int[3][]; |
| final int[] colors = new int[3]; |
| int i = 0; |
| |
| states[i] = FOCUSED_ENABLED_STATE_SET; |
| colors[i] = selectedColor; |
| i++; |
| |
| states[i] = PRESSED_ENABLED_STATE_SET; |
| colors[i] = selectedColor; |
| i++; |
| |
| // Default enabled state |
| states[i] = new int[0]; |
| colors[i] = Color.TRANSPARENT; |
| i++; |
| |
| return new ColorStateList(states, colors); |
| } |
| |
| private boolean shouldAnimateVisibilityChange() { |
| return ViewCompat.isLaidOut(mView) && !mView.isInEditMode(); |
| } |
| |
| private void updateFromViewRotation() { |
| if (Build.VERSION.SDK_INT == 19) { |
| // KitKat seems to have an issue with views which are rotated with angles which are |
| // not divisible by 90. Worked around by moving to software rendering in these cases. |
| if ((mRotation % 90) != 0) { |
| if (mView.getLayerType() != View.LAYER_TYPE_SOFTWARE) { |
| mView.setLayerType(View.LAYER_TYPE_SOFTWARE, null); |
| } |
| } else { |
| if (mView.getLayerType() != View.LAYER_TYPE_NONE) { |
| mView.setLayerType(View.LAYER_TYPE_NONE, null); |
| } |
| } |
| } |
| |
| // Offset any View rotation |
| if (mShadowDrawable != null) { |
| mShadowDrawable.setRotation(-mRotation); |
| } |
| if (mBorderDrawable != null) { |
| mBorderDrawable.setRotation(-mRotation); |
| } |
| } |
| } |