FloatingActionButton.java revision d95e0bb0271ae59387dd9d2ca402ad6f39d789d2
19840efe3dbdc7026521da8576574c55120782f6cChris Banes/*
29840efe3dbdc7026521da8576574c55120782f6cChris Banes * Copyright (C) 2015 The Android Open Source Project
39840efe3dbdc7026521da8576574c55120782f6cChris Banes *
49840efe3dbdc7026521da8576574c55120782f6cChris Banes * Licensed under the Apache License, Version 2.0 (the "License");
59840efe3dbdc7026521da8576574c55120782f6cChris Banes * you may not use this file except in compliance with the License.
69840efe3dbdc7026521da8576574c55120782f6cChris Banes * You may obtain a copy of the License at
79840efe3dbdc7026521da8576574c55120782f6cChris Banes *
89840efe3dbdc7026521da8576574c55120782f6cChris Banes *      http://www.apache.org/licenses/LICENSE-2.0
99840efe3dbdc7026521da8576574c55120782f6cChris Banes *
109840efe3dbdc7026521da8576574c55120782f6cChris Banes * Unless required by applicable law or agreed to in writing, software
119840efe3dbdc7026521da8576574c55120782f6cChris Banes * distributed under the License is distributed on an "AS IS" BASIS,
129840efe3dbdc7026521da8576574c55120782f6cChris Banes * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139840efe3dbdc7026521da8576574c55120782f6cChris Banes * See the License for the specific language governing permissions and
149840efe3dbdc7026521da8576574c55120782f6cChris Banes * limitations under the License.
159840efe3dbdc7026521da8576574c55120782f6cChris Banes */
169840efe3dbdc7026521da8576574c55120782f6cChris Banes
179840efe3dbdc7026521da8576574c55120782f6cChris Banespackage android.support.design.widget;
189840efe3dbdc7026521da8576574c55120782f6cChris Banes
199840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.annotation.TargetApi;
209840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.content.Context;
219840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.content.res.ColorStateList;
229840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.content.res.TypedArray;
239840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.graphics.PorterDuff;
249840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.graphics.Rect;
259840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.graphics.drawable.Drawable;
269840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.os.Build;
277a13c8489daca7915623dd673df49de2d1a0bf30Chris Banesimport android.support.annotation.ColorInt;
28d9cbe69a6661315238d856abc22578d03666f63bChris Banesimport android.support.annotation.NonNull;
299840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.support.annotation.Nullable;
309840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.support.design.R;
310ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Weiimport android.support.design.widget.FloatingActionButtonImpl.InternalVisibilityChangedListener;
32b7f9224b1495db47eb8fd813b5912250e900770aChris Banesimport android.support.v4.view.ViewCompat;
339840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.util.AttributeSet;
34097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banesimport android.util.Log;
3514d064edb3e4a16a3b90a4a850560177bea1e60dChris Banesimport android.view.View;
369840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.widget.ImageView;
379840efe3dbdc7026521da8576574c55120782f6cChris Banes
38a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banesimport java.util.List;
39b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
409840efe3dbdc7026521da8576574c55120782f6cChris Banes/**
419840efe3dbdc7026521da8576574c55120782f6cChris Banes * Floating action buttons are used for a special type of promoted action. They are distinguished
4214d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes * by a circled icon floating above the UI and have special motion behaviors related to morphing,
439840efe3dbdc7026521da8576574c55120782f6cChris Banes * launching, and the transferring anchor point.
449840efe3dbdc7026521da8576574c55120782f6cChris Banes *
459fb154338a62edc2c57dc036895199d6f1769400Chris Banes * <p>Floating action buttons come in two sizes: the default and the mini. The size can be
469fb154338a62edc2c57dc036895199d6f1769400Chris Banes * controlled with the {@code fabSize} attribute.</p>
479fb154338a62edc2c57dc036895199d6f1769400Chris Banes *
489fb154338a62edc2c57dc036895199d6f1769400Chris Banes * <p>As this class descends from {@link ImageView}, you can control the icon which is displayed
499fb154338a62edc2c57dc036895199d6f1769400Chris Banes * via {@link #setImageDrawable(Drawable)}.</p>
509fb154338a62edc2c57dc036895199d6f1769400Chris Banes *
519fb154338a62edc2c57dc036895199d6f1769400Chris Banes * <p>The background color of this view defaults to the your theme's {@code colorAccent}. If you
529fb154338a62edc2c57dc036895199d6f1769400Chris Banes * wish to change this at runtime then you can do so via
539fb154338a62edc2c57dc036895199d6f1769400Chris Banes * {@link #setBackgroundTintList(ColorStateList)}.</p>
549fb154338a62edc2c57dc036895199d6f1769400Chris Banes *
559fb154338a62edc2c57dc036895199d6f1769400Chris Banes * @attr ref android.support.design.R.styleable#FloatingActionButton_fabSize
569840efe3dbdc7026521da8576574c55120782f6cChris Banes */
57b7f9224b1495db47eb8fd813b5912250e900770aChris Banes@CoordinatorLayout.DefaultBehavior(FloatingActionButton.Behavior.class)
58fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banespublic class FloatingActionButton extends VisibilityAwareImageButton {
599840efe3dbdc7026521da8576574c55120782f6cChris Banes
60097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    private static final String LOG_TAG = "FloatingActionButton";
61097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes
620ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    /**
630ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * Callback to be invoked when the visibility of a FloatingActionButton changes.
640ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     */
650ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    public abstract static class OnVisibilityChangedListener {
660ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        /**
670ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * Called when a FloatingActionButton has been
680ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * {@link #show(OnVisibilityChangedListener) shown}.
690ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         *
700ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * @param fab the FloatingActionButton that was shown.
710ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         */
720ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        public void onShown(FloatingActionButton fab) {}
730ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
740ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        /**
750ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * Called when a FloatingActionButton has been
760ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * {@link #hide(OnVisibilityChangedListener) hidden}.
770ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         *
780ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         * @param fab the FloatingActionButton that was hidden.
790ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei         */
800ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        public void onHidden(FloatingActionButton fab) {}
810ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    }
820ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
839840efe3dbdc7026521da8576574c55120782f6cChris Banes    // These values must match those in the attrs declaration
849840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static final int SIZE_MINI = 1;
859840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static final int SIZE_NORMAL = 0;
869840efe3dbdc7026521da8576574c55120782f6cChris Banes
879840efe3dbdc7026521da8576574c55120782f6cChris Banes    private ColorStateList mBackgroundTint;
889840efe3dbdc7026521da8576574c55120782f6cChris Banes    private PorterDuff.Mode mBackgroundTintMode;
899840efe3dbdc7026521da8576574c55120782f6cChris Banes
90cd78954a2b32d9c22686f12c194fac7e49566cf6Chris Banes    private int mBorderWidth;
919840efe3dbdc7026521da8576574c55120782f6cChris Banes    private int mRippleColor;
929840efe3dbdc7026521da8576574c55120782f6cChris Banes    private int mSize;
93d9cbe69a6661315238d856abc22578d03666f63bChris Banes    private int mImagePadding;
949840efe3dbdc7026521da8576574c55120782f6cChris Banes
956d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    private boolean mCompatPadding;
96cdc736866534c604c4015c78371ade52bb6d52dfChris Banes    private final Rect mShadowPadding = new Rect();
979840efe3dbdc7026521da8576574c55120782f6cChris Banes
98d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes    private FloatingActionButtonImpl mImpl;
999840efe3dbdc7026521da8576574c55120782f6cChris Banes
1009840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context) {
1019840efe3dbdc7026521da8576574c55120782f6cChris Banes        this(context, null);
1029840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1039840efe3dbdc7026521da8576574c55120782f6cChris Banes
1049840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context, AttributeSet attrs) {
1059840efe3dbdc7026521da8576574c55120782f6cChris Banes        this(context, attrs, 0);
1069840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1079840efe3dbdc7026521da8576574c55120782f6cChris Banes
1089840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context, AttributeSet attrs, int defStyleAttr) {
1099840efe3dbdc7026521da8576574c55120782f6cChris Banes        super(context, attrs, defStyleAttr);
1109840efe3dbdc7026521da8576574c55120782f6cChris Banes
111809bb62055ad42b88f3a69308be222801b89fbd9Chris Banes        ThemeUtils.checkAppCompatTheme(context);
112809bb62055ad42b88f3a69308be222801b89fbd9Chris Banes
1139840efe3dbdc7026521da8576574c55120782f6cChris Banes        TypedArray a = context.obtainStyledAttributes(attrs,
1149840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton, defStyleAttr,
1159840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.style.Widget_Design_FloatingActionButton);
1169840efe3dbdc7026521da8576574c55120782f6cChris Banes        mBackgroundTint = a.getColorStateList(R.styleable.FloatingActionButton_backgroundTint);
1179840efe3dbdc7026521da8576574c55120782f6cChris Banes        mBackgroundTintMode = parseTintMode(a.getInt(
1189840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton_backgroundTintMode, -1), null);
1199840efe3dbdc7026521da8576574c55120782f6cChris Banes        mRippleColor = a.getColor(R.styleable.FloatingActionButton_rippleColor, 0);
1209840efe3dbdc7026521da8576574c55120782f6cChris Banes        mSize = a.getInt(R.styleable.FloatingActionButton_fabSize, SIZE_NORMAL);
121cd78954a2b32d9c22686f12c194fac7e49566cf6Chris Banes        mBorderWidth = a.getDimensionPixelSize(R.styleable.FloatingActionButton_borderWidth, 0);
1229840efe3dbdc7026521da8576574c55120782f6cChris Banes        final float elevation = a.getDimension(R.styleable.FloatingActionButton_elevation, 0f);
1239840efe3dbdc7026521da8576574c55120782f6cChris Banes        final float pressedTranslationZ = a.getDimension(
1249840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton_pressedTranslationZ, 0f);
1256d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes        mCompatPadding = a.getBoolean(R.styleable.FloatingActionButton_useCompatPadding, false);
1269840efe3dbdc7026521da8576574c55120782f6cChris Banes        a.recycle();
1279840efe3dbdc7026521da8576574c55120782f6cChris Banes
128d9cbe69a6661315238d856abc22578d03666f63bChris Banes        final int maxImageSize = (int) getResources().getDimension(R.dimen.design_fab_image_size);
129d9cbe69a6661315238d856abc22578d03666f63bChris Banes        mImagePadding = (getSizeDimension() - maxImageSize) / 2;
1309840efe3dbdc7026521da8576574c55120782f6cChris Banes
131d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().setBackgroundDrawable(mBackgroundTint, mBackgroundTintMode,
132097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes                mRippleColor, mBorderWidth);
133d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().setElevation(elevation);
134d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().setPressedTranslationZ(pressedTranslationZ);
135d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().updatePadding();
1369840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1379840efe3dbdc7026521da8576574c55120782f6cChris Banes
1389840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
1399840efe3dbdc7026521da8576574c55120782f6cChris Banes    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
1409840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int preferredSize = getSizeDimension();
1419840efe3dbdc7026521da8576574c55120782f6cChris Banes
1429840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int w = resolveAdjustedSize(preferredSize, widthMeasureSpec);
1439840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int h = resolveAdjustedSize(preferredSize, heightMeasureSpec);
1449840efe3dbdc7026521da8576574c55120782f6cChris Banes
1459840efe3dbdc7026521da8576574c55120782f6cChris Banes        // As we want to stay circular, we set both dimensions to be the
1469840efe3dbdc7026521da8576574c55120782f6cChris Banes        // smallest resolved dimension
1479840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int d = Math.min(w, h);
1489840efe3dbdc7026521da8576574c55120782f6cChris Banes
1499840efe3dbdc7026521da8576574c55120782f6cChris Banes        // We add the shadow's padding to the measured dimension
1509840efe3dbdc7026521da8576574c55120782f6cChris Banes        setMeasuredDimension(
1519840efe3dbdc7026521da8576574c55120782f6cChris Banes                d + mShadowPadding.left + mShadowPadding.right,
1529840efe3dbdc7026521da8576574c55120782f6cChris Banes                d + mShadowPadding.top + mShadowPadding.bottom);
1539840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1549840efe3dbdc7026521da8576574c55120782f6cChris Banes
1559840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1569840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Set the ripple color for this {@link FloatingActionButton}.
1579840efe3dbdc7026521da8576574c55120782f6cChris Banes     * <p>
1589840efe3dbdc7026521da8576574c55120782f6cChris Banes     * When running on devices with KitKat or below, we draw a fill rather than a ripple.
1599840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1609840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param color ARGB color to use for the ripple.
1616d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
1626d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @attr ref android.support.design.R.styleable#FloatingActionButton_rippleColor
1639840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1647a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes    public void setRippleColor(@ColorInt int color) {
1659840efe3dbdc7026521da8576574c55120782f6cChris Banes        if (mRippleColor != color) {
1669840efe3dbdc7026521da8576574c55120782f6cChris Banes            mRippleColor = color;
167d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            getImpl().setRippleColor(color);
1689840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
1699840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1709840efe3dbdc7026521da8576574c55120782f6cChris Banes
1719840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1729840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Return the tint applied to the background drawable, if specified.
1739840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1749840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @return the tint applied to the background drawable
1759840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @see #setBackgroundTintList(ColorStateList)
1769840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1779840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Nullable
1789840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
1799840efe3dbdc7026521da8576574c55120782f6cChris Banes    public ColorStateList getBackgroundTintList() {
1809840efe3dbdc7026521da8576574c55120782f6cChris Banes        return mBackgroundTint;
1819840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1829840efe3dbdc7026521da8576574c55120782f6cChris Banes
1839840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1849840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Applies a tint to the background drawable. Does not modify the current tint
1859840efe3dbdc7026521da8576574c55120782f6cChris Banes     * mode, which is {@link PorterDuff.Mode#SRC_IN} by default.
1869840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1879840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param tint the tint to apply, may be {@code null} to clear tint
1889840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1899840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setBackgroundTintList(@Nullable ColorStateList tint) {
1907a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes        if (mBackgroundTint != tint) {
1917a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes            mBackgroundTint = tint;
192d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            getImpl().setBackgroundTintList(tint);
1937a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes        }
1949840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1959840efe3dbdc7026521da8576574c55120782f6cChris Banes
1969840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1979840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Return the blending mode used to apply the tint to the background
1989840efe3dbdc7026521da8576574c55120782f6cChris Banes     * drawable, if specified.
1999840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
2009840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @return the blending mode used to apply the tint to the background
2019840efe3dbdc7026521da8576574c55120782f6cChris Banes     *         drawable
2029840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @see #setBackgroundTintMode(PorterDuff.Mode)
2039840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
2049840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Nullable
2059840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
2069840efe3dbdc7026521da8576574c55120782f6cChris Banes    public PorterDuff.Mode getBackgroundTintMode() {
2079840efe3dbdc7026521da8576574c55120782f6cChris Banes        return mBackgroundTintMode;
2089840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2099840efe3dbdc7026521da8576574c55120782f6cChris Banes
2109840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
2119840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Specifies the blending mode used to apply the tint specified by
2129840efe3dbdc7026521da8576574c55120782f6cChris Banes     * {@link #setBackgroundTintList(ColorStateList)}} to the background
2139840efe3dbdc7026521da8576574c55120782f6cChris Banes     * drawable. The default mode is {@link PorterDuff.Mode#SRC_IN}.
2149840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
2159840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param tintMode the blending mode used to apply the tint, may be
2169840efe3dbdc7026521da8576574c55120782f6cChris Banes     *                 {@code null} to clear tint
2179840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
2189840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setBackgroundTintMode(@Nullable PorterDuff.Mode tintMode) {
2197a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes        if (mBackgroundTintMode != tintMode) {
2207a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes            mBackgroundTintMode = tintMode;
221d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            getImpl().setBackgroundTintMode(tintMode);
2227a13c8489daca7915623dd673df49de2d1a0bf30Chris Banes        }
2239840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2249840efe3dbdc7026521da8576574c55120782f6cChris Banes
2259840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
226097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    public void setBackgroundDrawable(Drawable background) {
227097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes        Log.i(LOG_TAG, "Setting a custom background is not supported.");
228097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    }
229097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes
230097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    @Override
231097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    public void setBackgroundResource(int resid) {
232097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes        Log.i(LOG_TAG, "Setting a custom background is not supported.");
233097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    }
234097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes
235097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    @Override
236097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes    public void setBackgroundColor(int color) {
237097e80a3c5518c6bf2e9f3f9b55ed9f4b5cc37e8Chris Banes        Log.i(LOG_TAG, "Setting a custom background is not supported.");
238be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    }
239be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes
240be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    /**
241be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes     * Shows the button.
2428c05e5f52fbc790b745e768398d9e69d6b9d9ee1Chris Banes     * <p>This method will animate the button show if the view has already been laid out.</p>
243be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes     */
244be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    public void show() {
245fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes        show(null);
2460ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    }
2470ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
2480ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    /**
2490ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * Shows the button.
2508c05e5f52fbc790b745e768398d9e69d6b9d9ee1Chris Banes     * <p>This method will animate the button show if the view has already been laid out.</p>
2510ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     *
2520ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * @param listener the listener to notify when this view is shown
2530ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     */
2540ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    public void show(@Nullable final OnVisibilityChangedListener listener) {
255fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes        show(listener, true);
256fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes    }
257fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes
258fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes    private void show(OnVisibilityChangedListener listener, boolean fromUser) {
259d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().show(wrapOnVisibilityChangedListener(listener), fromUser);
260be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    }
261be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes
262be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    /**
263be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes     * Hides the button.
264be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes     * <p>This method will animate the button hide if the view has already been laid out.</p>
265be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes     */
266be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes    public void hide() {
267fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes        hide(null);
2680ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    }
2690ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
2700ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    /**
2710ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * Hides the button.
2720ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * <p>This method will animate the button hide if the view has already been laid out.</p>
2730ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     *
2740ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     * @param listener the listener to notify when this view is hidden
2750ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei     */
2760ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    public void hide(@Nullable OnVisibilityChangedListener listener) {
277fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes        hide(listener, true);
278fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes    }
279fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes
280fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes    private void hide(@Nullable OnVisibilityChangedListener listener, boolean fromUser) {
281d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().hide(wrapOnVisibilityChangedListener(listener), fromUser);
2820ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    }
2830ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
2846d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    /**
2856d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * Set whether FloatingActionButton should add inner padding on platforms Lollipop and after,
2866d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * to ensure consistent dimensions on all platforms.
2876d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
2886d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @param useCompatPadding true if FloatingActionButton is adding inner padding on platforms
2896d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *                         Lollipop and after, to ensure consistent dimensions on all platforms.
2906d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
2916d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @attr ref android.support.design.R.styleable#FloatingActionButton_useCompatPadding
2926d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @see #getUseCompatPadding()
2936d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     */
2946d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    public void setUseCompatPadding(boolean useCompatPadding) {
2956d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes        if (mCompatPadding != useCompatPadding) {
2966d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes            mCompatPadding = useCompatPadding;
297d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            getImpl().onCompatShadowChanged();
2986d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes        }
2996d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    }
3006d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes
3016d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    /**
3026d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * Returns whether FloatingActionButton will add inner padding on platforms Lollipop and after.
3036d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
3046d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @return true if FloatingActionButton is adding inner padding on platforms Lollipop and after,
3056d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * to ensure consistent dimensions on all platforms.
3066d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
3076d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @attr ref android.support.design.R.styleable#FloatingActionButton_useCompatPadding
3086d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @see #setUseCompatPadding(boolean)
3096d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     */
3106d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    public boolean getUseCompatPadding() {
3116d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes        return mCompatPadding;
3126d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    }
3136d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes
3140ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    @Nullable
3150ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei    private InternalVisibilityChangedListener wrapOnVisibilityChangedListener(
3160ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            @Nullable final OnVisibilityChangedListener listener) {
3170ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        if (listener == null) {
3180ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            return null;
3190ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        }
3200ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
3210ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        return new InternalVisibilityChangedListener() {
3220ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            @Override
3230ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            public void onShown() {
3240ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei                listener.onShown(FloatingActionButton.this);
3250ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            }
3260ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei
3270ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            @Override
3280ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            public void onHidden() {
3290ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei                listener.onHidden(FloatingActionButton.this);
3300ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei            }
3310ad7ef59b28d8ffafd551d2756b5a8ec47c90682Mark Wei        };
3329840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
3339840efe3dbdc7026521da8576574c55120782f6cChris Banes
3349840efe3dbdc7026521da8576574c55120782f6cChris Banes    final int getSizeDimension() {
3359840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (mSize) {
3369840efe3dbdc7026521da8576574c55120782f6cChris Banes            case SIZE_MINI:
337a577676a64e5353b8ec927117151aa6be84adf66Chris Banes                return getResources().getDimensionPixelSize(R.dimen.design_fab_size_mini);
3389840efe3dbdc7026521da8576574c55120782f6cChris Banes            case SIZE_NORMAL:
3399840efe3dbdc7026521da8576574c55120782f6cChris Banes            default:
340a577676a64e5353b8ec927117151aa6be84adf66Chris Banes                return getResources().getDimensionPixelSize(R.dimen.design_fab_size_normal);
3419840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
3429840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
3439840efe3dbdc7026521da8576574c55120782f6cChris Banes
3449840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
345d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    protected void onAttachedToWindow() {
346d9770e12c8ff2d4417700492c6616572be897e93Chris Banes        super.onAttachedToWindow();
347d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().onAttachedToWindow();
348d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    }
349d9770e12c8ff2d4417700492c6616572be897e93Chris Banes
350d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    @Override
351d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    protected void onDetachedFromWindow() {
352d9770e12c8ff2d4417700492c6616572be897e93Chris Banes        super.onDetachedFromWindow();
353d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().onDetachedFromWindow();
354d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    }
355d9770e12c8ff2d4417700492c6616572be897e93Chris Banes
356d9770e12c8ff2d4417700492c6616572be897e93Chris Banes    @Override
3579840efe3dbdc7026521da8576574c55120782f6cChris Banes    protected void drawableStateChanged() {
3589840efe3dbdc7026521da8576574c55120782f6cChris Banes        super.drawableStateChanged();
359d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().onDrawableStateChanged(getDrawableState());
3609840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
3619840efe3dbdc7026521da8576574c55120782f6cChris Banes
3629840efe3dbdc7026521da8576574c55120782f6cChris Banes    @TargetApi(Build.VERSION_CODES.HONEYCOMB)
3639840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
3649840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void jumpDrawablesToCurrentState() {
3659840efe3dbdc7026521da8576574c55120782f6cChris Banes        super.jumpDrawablesToCurrentState();
366d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().jumpDrawableToCurrentState();
3679840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
3689840efe3dbdc7026521da8576574c55120782f6cChris Banes
369d9cbe69a6661315238d856abc22578d03666f63bChris Banes    /**
370d9cbe69a6661315238d856abc22578d03666f63bChris Banes     * Return in {@code rect} the bounds of the actual floating action button content in view-local
371d9cbe69a6661315238d856abc22578d03666f63bChris Banes     * coordinates. This is defined as anything within any visible shadow.
372d9cbe69a6661315238d856abc22578d03666f63bChris Banes     *
373d9cbe69a6661315238d856abc22578d03666f63bChris Banes     * @return true if this view actually has been laid out and has a content rect, else false.
374d9cbe69a6661315238d856abc22578d03666f63bChris Banes     */
375d9cbe69a6661315238d856abc22578d03666f63bChris Banes    public boolean getContentRect(@NonNull Rect rect) {
376d9cbe69a6661315238d856abc22578d03666f63bChris Banes        if (ViewCompat.isLaidOut(this)) {
377d9cbe69a6661315238d856abc22578d03666f63bChris Banes            rect.set(0, 0, getWidth(), getHeight());
378d9cbe69a6661315238d856abc22578d03666f63bChris Banes            rect.left += mShadowPadding.left;
379d9cbe69a6661315238d856abc22578d03666f63bChris Banes            rect.top += mShadowPadding.top;
380d9cbe69a6661315238d856abc22578d03666f63bChris Banes            rect.right -= mShadowPadding.right;
381d9cbe69a6661315238d856abc22578d03666f63bChris Banes            rect.bottom -= mShadowPadding.bottom;
382d9cbe69a6661315238d856abc22578d03666f63bChris Banes            return true;
383d9cbe69a6661315238d856abc22578d03666f63bChris Banes        } else {
384d9cbe69a6661315238d856abc22578d03666f63bChris Banes            return false;
385d9cbe69a6661315238d856abc22578d03666f63bChris Banes        }
386d9cbe69a6661315238d856abc22578d03666f63bChris Banes    }
387d9cbe69a6661315238d856abc22578d03666f63bChris Banes
3883d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes    /**
3893d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes     * Returns the FloatingActionButton's background, minus any compatible shadow implementation.
3903d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes     */
3913d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes    @NonNull
3923d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes    public Drawable getContentBackground() {
393d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        return getImpl().getContentBackground();
3943d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes    }
3953d81c900316412b4130bf40e0dd8b0d3d3a93e78Chris Banes
3969840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static int resolveAdjustedSize(int desiredSize, int measureSpec) {
3979840efe3dbdc7026521da8576574c55120782f6cChris Banes        int result = desiredSize;
3989840efe3dbdc7026521da8576574c55120782f6cChris Banes        int specMode = MeasureSpec.getMode(measureSpec);
3999840efe3dbdc7026521da8576574c55120782f6cChris Banes        int specSize = MeasureSpec.getSize(measureSpec);
4009840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (specMode) {
4019840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.UNSPECIFIED:
4029840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Parent says we can be as big as we want. Just don't be larger
4039840efe3dbdc7026521da8576574c55120782f6cChris Banes                // than max size imposed on ourselves.
4049840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = desiredSize;
4059840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
4069840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.AT_MOST:
4079840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Parent says we can be as big as we want, up to specSize.
4089840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Don't be larger than specSize, and don't be larger than
4099840efe3dbdc7026521da8576574c55120782f6cChris Banes                // the max size imposed on ourselves.
4109840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = Math.min(desiredSize, specSize);
4119840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
4129840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.EXACTLY:
4139840efe3dbdc7026521da8576574c55120782f6cChris Banes                // No choice. Do what we are told.
4149840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = specSize;
4159840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
4169840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
4179840efe3dbdc7026521da8576574c55120782f6cChris Banes        return result;
4189840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
4199840efe3dbdc7026521da8576574c55120782f6cChris Banes
4209840efe3dbdc7026521da8576574c55120782f6cChris Banes    static PorterDuff.Mode parseTintMode(int value, PorterDuff.Mode defaultMode) {
4219840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (value) {
4229840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 3:
4239840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_OVER;
4249840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 5:
4259840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_IN;
4269840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 9:
4279840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_ATOP;
4289840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 14:
4299840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.MULTIPLY;
4309840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 15:
4319840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SCREEN;
4329840efe3dbdc7026521da8576574c55120782f6cChris Banes            default:
4339840efe3dbdc7026521da8576574c55120782f6cChris Banes                return defaultMode;
4349840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
4359840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
43614d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes
43714d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes    /**
438b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * Behavior designed for use with {@link FloatingActionButton} instances. It's main function
439b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * is to move {@link FloatingActionButton} views so that any displayed {@link Snackbar}s do
440b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * not cover them.
441b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     */
442b7f9224b1495db47eb8fd813b5912250e900770aChris Banes    public static class Behavior extends CoordinatorLayout.Behavior<FloatingActionButton> {
443b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        // We only support the FAB <> Snackbar shift movement on Honeycomb and above. This is
444b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        // because we can use view translation properties which greatly simplifies the code.
445b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        private static final boolean SNACKBAR_BEHAVIOR_ENABLED = Build.VERSION.SDK_INT >= 11;
446b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
447a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes        private ValueAnimatorCompat mFabTranslationYAnimator;
44818d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes        private float mFabTranslationY;
449a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private Rect mTmpRect;
450b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
451b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        @Override
452b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        public boolean layoutDependsOn(CoordinatorLayout parent,
453be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                FloatingActionButton child, View dependency) {
454a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            // We're dependent on all SnackbarLayouts (if enabled)
455a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            return SNACKBAR_BEHAVIOR_ENABLED && dependency instanceof Snackbar.SnackbarLayout;
456b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
457b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
458b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        @Override
459b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        public boolean onDependentViewChanged(CoordinatorLayout parent, FloatingActionButton child,
460a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                View dependency) {
461a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            if (dependency instanceof Snackbar.SnackbarLayout) {
462a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                updateFabTranslationForSnackbar(parent, child, dependency);
463a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            } else if (dependency instanceof AppBarLayout) {
464be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // If we're depending on an AppBarLayout we will show/hide it automatically
465be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // if the FAB is anchored to the AppBarLayout
466be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                updateFabVisibility(parent, (AppBarLayout) dependency, child);
467be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            }
468be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            return false;
469be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes        }
470a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
471be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes        private boolean updateFabVisibility(CoordinatorLayout parent,
472be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                AppBarLayout appBarLayout, FloatingActionButton child) {
473be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            final CoordinatorLayout.LayoutParams lp =
474be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                    (CoordinatorLayout.LayoutParams) child.getLayoutParams();
475be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            if (lp.getAnchorId() != appBarLayout.getId()) {
476be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // The anchor ID doesn't match the dependency, so we won't automatically
477be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // show/hide the FAB
478be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                return false;
479be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            }
480a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
481fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes            if (child.getUserSetVisibility() != VISIBLE) {
482fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes                // The view isn't set to be visible so skip changing it's visibility
483fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes                return false;
484fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes            }
485fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes
486be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            if (mTmpRect == null) {
487be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                mTmpRect = new Rect();
488a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            }
489be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes
490be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            // First, let's get the visible rect of the dependency
491be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            final Rect rect = mTmpRect;
492be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            ViewGroupUtils.getDescendantRect(parent, appBarLayout, rect);
493be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes
494be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            if (rect.bottom <= appBarLayout.getMinimumHeightForVisibleOverlappingContent()) {
495be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // If the anchor's bottom is below the seam, we'll animate our FAB out
496fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes                child.hide(null, false);
497be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            } else {
498be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                // Else, we'll animate our FAB back in
499fc780bab91bd4275ae2c3b75c3dfb327e008e4dbChris Banes                child.show(null, false);
500be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            }
501be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            return true;
502b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
503b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
504a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private void updateFabTranslationForSnackbar(CoordinatorLayout parent,
505a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                final FloatingActionButton fab, View snackbar) {
50618d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            final float targetTransY = getFabTranslationYForSnackbar(parent, fab);
50718d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            if (mFabTranslationY == targetTransY) {
50818d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                // We're already at (or currently animating to) the target value, return...
50918d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                return;
51018d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            }
51118d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes
51218d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            final float currentTransY = ViewCompat.getTranslationY(fab);
51318d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes
514a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes            // Make sure that any current animation is cancelled
515a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes            if (mFabTranslationYAnimator != null && mFabTranslationYAnimator.isRunning()) {
516a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                mFabTranslationYAnimator.cancel();
517a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes            }
518a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes
519e33473d0e35315c02243363a7479a2c361765751Chris Banes            if (fab.isShown()
520e33473d0e35315c02243363a7479a2c361765751Chris Banes                    && Math.abs(currentTransY - targetTransY) > (fab.getHeight() * 0.667f)) {
52118d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                // If the FAB will be travelling by more than 2/3 of it's height, let's animate
52218d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                // it instead
523a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                if (mFabTranslationYAnimator == null) {
524a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                    mFabTranslationYAnimator = ViewUtils.createAnimator();
525a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                    mFabTranslationYAnimator.setInterpolator(
526a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                            AnimationUtils.FAST_OUT_SLOW_IN_INTERPOLATOR);
527a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                    mFabTranslationYAnimator.setUpdateListener(
528a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                            new ValueAnimatorCompat.AnimatorUpdateListener() {
529a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                                @Override
530a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                                public void onAnimationUpdate(ValueAnimatorCompat animator) {
531a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                                    ViewCompat.setTranslationY(fab,
532a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                                            animator.getAnimatedFloatValue());
533a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                                }
534a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                            });
535a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                }
536a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                mFabTranslationYAnimator.setFloatValues(currentTransY, targetTransY);
537a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes                mFabTranslationYAnimator.start();
53818d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            } else {
53918d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                // Now update the translation Y
54018d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes                ViewCompat.setTranslationY(fab, targetTransY);
54118d22257ccfb5cebb3ccd2450736e735ed1fb9bbChris Banes            }
542a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes
543a419ee1ef9aef8b567f1ccd8c29d01ec7bff4cc9Chris Banes            mFabTranslationY = targetTransY;
544b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
545b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
546a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private float getFabTranslationYForSnackbar(CoordinatorLayout parent,
547a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                FloatingActionButton fab) {
548b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            float minOffset = 0;
549a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            final List<View> dependencies = parent.getDependencies(fab);
550a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            for (int i = 0, z = dependencies.size(); i < z; i++) {
551a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                final View view = dependencies.get(i);
552a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                if (view instanceof Snackbar.SnackbarLayout && parent.doViewsOverlap(fab, view)) {
553a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    minOffset = Math.min(minOffset,
554a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            ViewCompat.getTranslationY(view) - view.getHeight());
555b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                }
556b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            }
557a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
558b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            return minOffset;
559b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
560b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
561e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        @Override
562e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        public boolean onLayoutChild(CoordinatorLayout parent, FloatingActionButton child,
563e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                int layoutDirection) {
564be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            // First, lets make sure that the visibility of the FAB is consistent
565be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            final List<View> dependencies = parent.getDependencies(child);
566be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            for (int i = 0, count = dependencies.size(); i < count; i++) {
567be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                final View dependency = dependencies.get(i);
568be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                if (dependency instanceof AppBarLayout
569be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                        && updateFabVisibility(parent, (AppBarLayout) dependency, child)) {
570be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                    break;
571be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes                }
572be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            }
573be48ed9161c09c4b2178ab6dbe28638222809fc7Chris Banes            // Now let the CoordinatorLayout lay out the FAB
574e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            parent.onLayoutChild(child, layoutDirection);
575e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            // Now offset it if needed
576e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            offsetIfNeeded(parent, child);
577e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            return true;
578e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        }
579e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes
580e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        /**
581e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes         * Pre-Lollipop we use padding so that the shadow has enough space to be drawn. This method
582e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes         * offsets our layout position so that we're positioned correctly if we're on one of
583e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes         * our parent's edges.
584e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes         */
585e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        private void offsetIfNeeded(CoordinatorLayout parent, FloatingActionButton fab) {
586e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            final Rect padding = fab.mShadowPadding;
587e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes
588e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            if (padding != null && padding.centerX() > 0 && padding.centerY() > 0) {
589e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                final CoordinatorLayout.LayoutParams lp =
590e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                        (CoordinatorLayout.LayoutParams) fab.getLayoutParams();
591e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes
592e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                int offsetTB = 0, offsetLR = 0;
593e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes
594e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                if (fab.getRight() >= parent.getWidth() - lp.rightMargin) {
595e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    // If we're on the left edge, shift it the right
596e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    offsetLR = padding.right;
597e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                } else if (fab.getLeft() <= lp.leftMargin) {
598e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    // If we're on the left edge, shift it the left
599e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    offsetLR = -padding.left;
600e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                }
601e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                if (fab.getBottom() >= parent.getBottom() - lp.bottomMargin) {
602e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    // If we're on the bottom edge, shift it down
603e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    offsetTB = padding.bottom;
604e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                } else if (fab.getTop() <= lp.topMargin) {
605e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    // If we're on the top edge, shift it up
606e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                    offsetTB = -padding.top;
607e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                }
608e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes
609e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                fab.offsetTopAndBottom(offsetTB);
610e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes                fab.offsetLeftAndRight(offsetLR);
611e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes            }
612e882ef3492de3d2bb687b454e08b870b06d8f4e2Chris Banes        }
613b7f9224b1495db47eb8fd813b5912250e900770aChris Banes    }
6146d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes
6156d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    /**
6166d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * Returns the backward compatible elevation of the FloatingActionButton.
6176d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
6182b1d1d93a93070601d3894f523d6421c64544246Chris Banes     * @return the backward compatible elevation in pixels.
6196d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @attr ref android.support.design.R.styleable#FloatingActionButton_elevation
6201e220ff878c4b9c22aff3a6afc20aa89449c1833Chris Banes     * @see #setCompatElevation(float)
6216d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     */
6221e220ff878c4b9c22aff3a6afc20aa89449c1833Chris Banes    public float getCompatElevation() {
623d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        return getImpl().getElevation();
6246d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    }
6256d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes
6266d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    /**
6276d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * Updates the backward compatible elevation of the FloatingActionButton.
6286d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     *
6296d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @param elevation The backward compatible elevation in pixels.
6306d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @attr ref android.support.design.R.styleable#FloatingActionButton_elevation
6311e220ff878c4b9c22aff3a6afc20aa89449c1833Chris Banes     * @see #getCompatElevation()
6326d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     * @see #setUseCompatPadding(boolean)
6336d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes     */
6341e220ff878c4b9c22aff3a6afc20aa89449c1833Chris Banes    public void setCompatElevation(float elevation) {
635d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        getImpl().setElevation(elevation);
6366d7a9a02765e4cb497081e66dafb5d9fa76f4312Chris Banes    }
637cdc736866534c604c4015c78371ade52bb6d52dfChris Banes
638d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes    private FloatingActionButtonImpl getImpl() {
639d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        if (mImpl == null) {
640d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            mImpl = createImpl();
641d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        }
642d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes        return mImpl;
643d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes    }
644d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes
645d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes    private FloatingActionButtonImpl createImpl() {
646cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        final int sdk = Build.VERSION.SDK_INT;
647cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        if (sdk >= 21) {
648d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            return new FloatingActionButtonLollipop(this, new ShadowDelegateImpl());
649cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        } else if (sdk >= 14) {
650d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            return new FloatingActionButtonIcs(this, new ShadowDelegateImpl());
651cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        } else {
652d95e0bb0271ae59387dd9d2ca402ad6f39d789d2Chris Banes            return new FloatingActionButtonEclairMr1(this, new ShadowDelegateImpl());
653cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        }
654cdc736866534c604c4015c78371ade52bb6d52dfChris Banes    }
655cdc736866534c604c4015c78371ade52bb6d52dfChris Banes
656cdc736866534c604c4015c78371ade52bb6d52dfChris Banes    private class ShadowDelegateImpl implements ShadowViewDelegate {
657cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        @Override
658cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        public float getRadius() {
659cdc736866534c604c4015c78371ade52bb6d52dfChris Banes            return getSizeDimension() / 2f;
660cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        }
661cdc736866534c604c4015c78371ade52bb6d52dfChris Banes
662cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        @Override
663cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        public void setShadowPadding(int left, int top, int right, int bottom) {
664cdc736866534c604c4015c78371ade52bb6d52dfChris Banes            mShadowPadding.set(left, top, right, bottom);
665cdc736866534c604c4015c78371ade52bb6d52dfChris Banes            setPadding(left + mImagePadding, top + mImagePadding,
666cdc736866534c604c4015c78371ade52bb6d52dfChris Banes                    right + mImagePadding, bottom + mImagePadding);
667cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        }
668cdc736866534c604c4015c78371ade52bb6d52dfChris Banes
669cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        @Override
670cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        public void setBackgroundDrawable(Drawable background) {
671cdc736866534c604c4015c78371ade52bb6d52dfChris Banes            FloatingActionButton.super.setBackgroundDrawable(background);
672cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        }
673cdc736866534c604c4015c78371ade52bb6d52dfChris Banes
674cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        @Override
675cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        public boolean isCompatPaddingEnabled() {
676cdc736866534c604c4015c78371ade52bb6d52dfChris Banes            return mCompatPadding;
677cdc736866534c604c4015c78371ade52bb6d52dfChris Banes        }
678cdc736866534c604c4015c78371ade52bb6d52dfChris Banes    }
6799840efe3dbdc7026521da8576574c55120782f6cChris Banes}
680