FloatingActionButton.java revision a6a508b2296730ca6954aaebcca52a9962a5cb55
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;
279840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.support.annotation.Nullable;
289840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.support.design.R;
29b7f9224b1495db47eb8fd813b5912250e900770aChris Banesimport android.support.v4.view.ViewCompat;
30a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banesimport android.support.v4.view.ViewPropertyAnimatorListener;
319840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.util.AttributeSet;
3214d064edb3e4a16a3b90a4a850560177bea1e60dChris Banesimport android.view.View;
339840efe3dbdc7026521da8576574c55120782f6cChris Banesimport android.widget.ImageView;
349840efe3dbdc7026521da8576574c55120782f6cChris Banes
35a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banesimport java.util.List;
36b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
379840efe3dbdc7026521da8576574c55120782f6cChris Banes/**
389840efe3dbdc7026521da8576574c55120782f6cChris Banes * Floating action buttons are used for a special type of promoted action. They are distinguished
3914d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes * by a circled icon floating above the UI and have special motion behaviors related to morphing,
409840efe3dbdc7026521da8576574c55120782f6cChris Banes * launching, and the transferring anchor point.
419840efe3dbdc7026521da8576574c55120782f6cChris Banes *
429840efe3dbdc7026521da8576574c55120782f6cChris Banes * Floating action buttons come in two sizes: the default, which should be used in most cases, and
439840efe3dbdc7026521da8576574c55120782f6cChris Banes * the mini, which should only be used to create visual continuity with other elements on the
449840efe3dbdc7026521da8576574c55120782f6cChris Banes * screen.
459840efe3dbdc7026521da8576574c55120782f6cChris Banes */
46b7f9224b1495db47eb8fd813b5912250e900770aChris Banes@CoordinatorLayout.DefaultBehavior(FloatingActionButton.Behavior.class)
479840efe3dbdc7026521da8576574c55120782f6cChris Banespublic class FloatingActionButton extends ImageView {
489840efe3dbdc7026521da8576574c55120782f6cChris Banes
499840efe3dbdc7026521da8576574c55120782f6cChris Banes    // These values must match those in the attrs declaration
509840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static final int SIZE_MINI = 1;
519840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static final int SIZE_NORMAL = 0;
529840efe3dbdc7026521da8576574c55120782f6cChris Banes
539840efe3dbdc7026521da8576574c55120782f6cChris Banes    private ColorStateList mBackgroundTint;
549840efe3dbdc7026521da8576574c55120782f6cChris Banes    private PorterDuff.Mode mBackgroundTintMode;
559840efe3dbdc7026521da8576574c55120782f6cChris Banes
569840efe3dbdc7026521da8576574c55120782f6cChris Banes    private int mRippleColor;
579840efe3dbdc7026521da8576574c55120782f6cChris Banes    private int mSize;
589840efe3dbdc7026521da8576574c55120782f6cChris Banes    private int mContentPadding;
599840efe3dbdc7026521da8576574c55120782f6cChris Banes
609840efe3dbdc7026521da8576574c55120782f6cChris Banes    private final Rect mShadowPadding;
619840efe3dbdc7026521da8576574c55120782f6cChris Banes
629840efe3dbdc7026521da8576574c55120782f6cChris Banes    private final FloatingActionButtonImpl mImpl;
639840efe3dbdc7026521da8576574c55120782f6cChris Banes
649840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context) {
659840efe3dbdc7026521da8576574c55120782f6cChris Banes        this(context, null);
669840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
679840efe3dbdc7026521da8576574c55120782f6cChris Banes
689840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context, AttributeSet attrs) {
699840efe3dbdc7026521da8576574c55120782f6cChris Banes        this(context, attrs, 0);
709840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
719840efe3dbdc7026521da8576574c55120782f6cChris Banes
729840efe3dbdc7026521da8576574c55120782f6cChris Banes    public FloatingActionButton(Context context, AttributeSet attrs, int defStyleAttr) {
739840efe3dbdc7026521da8576574c55120782f6cChris Banes        super(context, attrs, defStyleAttr);
749840efe3dbdc7026521da8576574c55120782f6cChris Banes
759840efe3dbdc7026521da8576574c55120782f6cChris Banes        mShadowPadding = new Rect();
769840efe3dbdc7026521da8576574c55120782f6cChris Banes
779840efe3dbdc7026521da8576574c55120782f6cChris Banes        TypedArray a = context.obtainStyledAttributes(attrs,
789840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton, defStyleAttr,
799840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.style.Widget_Design_FloatingActionButton);
809840efe3dbdc7026521da8576574c55120782f6cChris Banes        Drawable background = a.getDrawable(R.styleable.FloatingActionButton_android_background);
819840efe3dbdc7026521da8576574c55120782f6cChris Banes        mBackgroundTint = a.getColorStateList(R.styleable.FloatingActionButton_backgroundTint);
829840efe3dbdc7026521da8576574c55120782f6cChris Banes        mBackgroundTintMode = parseTintMode(a.getInt(
839840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton_backgroundTintMode, -1), null);
849840efe3dbdc7026521da8576574c55120782f6cChris Banes        mRippleColor = a.getColor(R.styleable.FloatingActionButton_rippleColor, 0);
859840efe3dbdc7026521da8576574c55120782f6cChris Banes        mSize = a.getInt(R.styleable.FloatingActionButton_fabSize, SIZE_NORMAL);
869840efe3dbdc7026521da8576574c55120782f6cChris Banes        final float elevation = a.getDimension(R.styleable.FloatingActionButton_elevation, 0f);
879840efe3dbdc7026521da8576574c55120782f6cChris Banes        final float pressedTranslationZ = a.getDimension(
889840efe3dbdc7026521da8576574c55120782f6cChris Banes                R.styleable.FloatingActionButton_pressedTranslationZ, 0f);
899840efe3dbdc7026521da8576574c55120782f6cChris Banes        a.recycle();
909840efe3dbdc7026521da8576574c55120782f6cChris Banes
919840efe3dbdc7026521da8576574c55120782f6cChris Banes        final ShadowViewDelegate delegate = new ShadowViewDelegate() {
929840efe3dbdc7026521da8576574c55120782f6cChris Banes            @Override
939840efe3dbdc7026521da8576574c55120782f6cChris Banes            public float getRadius() {
949840efe3dbdc7026521da8576574c55120782f6cChris Banes                return getSizeDimension() / 2f;
959840efe3dbdc7026521da8576574c55120782f6cChris Banes            }
969840efe3dbdc7026521da8576574c55120782f6cChris Banes
979840efe3dbdc7026521da8576574c55120782f6cChris Banes            @Override
989840efe3dbdc7026521da8576574c55120782f6cChris Banes            public void setShadowPadding(int left, int top, int right, int bottom) {
999840efe3dbdc7026521da8576574c55120782f6cChris Banes                mShadowPadding.set(left, top, right, bottom);
1009840efe3dbdc7026521da8576574c55120782f6cChris Banes
1019840efe3dbdc7026521da8576574c55120782f6cChris Banes                setPadding(left + mContentPadding, top + mContentPadding,
1029840efe3dbdc7026521da8576574c55120782f6cChris Banes                        right + mContentPadding, bottom + mContentPadding);
1039840efe3dbdc7026521da8576574c55120782f6cChris Banes            }
1049840efe3dbdc7026521da8576574c55120782f6cChris Banes
1059840efe3dbdc7026521da8576574c55120782f6cChris Banes            @Override
1069840efe3dbdc7026521da8576574c55120782f6cChris Banes            public void setBackgroundDrawable(Drawable background) {
1079840efe3dbdc7026521da8576574c55120782f6cChris Banes                FloatingActionButton.super.setBackgroundDrawable(background);
1089840efe3dbdc7026521da8576574c55120782f6cChris Banes            }
1099840efe3dbdc7026521da8576574c55120782f6cChris Banes        };
1109840efe3dbdc7026521da8576574c55120782f6cChris Banes
1119840efe3dbdc7026521da8576574c55120782f6cChris Banes        if (Build.VERSION.SDK_INT >= 21) {
1129840efe3dbdc7026521da8576574c55120782f6cChris Banes            mImpl = new FloatingActionButtonLollipop(this, delegate);
1139840efe3dbdc7026521da8576574c55120782f6cChris Banes        } else {
1149840efe3dbdc7026521da8576574c55120782f6cChris Banes            mImpl = new FloatingActionButtonEclairMr1(this, delegate);
1159840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
1169840efe3dbdc7026521da8576574c55120782f6cChris Banes
1179840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int maxContentSize = (int) getResources().getDimension(R.dimen.fab_content_size);
1189840efe3dbdc7026521da8576574c55120782f6cChris Banes        mContentPadding = (getSizeDimension() - maxContentSize) / 2;
1199840efe3dbdc7026521da8576574c55120782f6cChris Banes
1209840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setBackgroundDrawable(background, mBackgroundTint,
1219840efe3dbdc7026521da8576574c55120782f6cChris Banes                mBackgroundTintMode, mRippleColor);
1229840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setElevation(elevation);
1239840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setPressedTranslationZ(pressedTranslationZ);
1249840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1259840efe3dbdc7026521da8576574c55120782f6cChris Banes
1269840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
1279840efe3dbdc7026521da8576574c55120782f6cChris Banes    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
1289840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int preferredSize = getSizeDimension();
1299840efe3dbdc7026521da8576574c55120782f6cChris Banes
1309840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int w = resolveAdjustedSize(preferredSize, widthMeasureSpec);
1319840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int h = resolveAdjustedSize(preferredSize, heightMeasureSpec);
1329840efe3dbdc7026521da8576574c55120782f6cChris Banes
1339840efe3dbdc7026521da8576574c55120782f6cChris Banes        // As we want to stay circular, we set both dimensions to be the
1349840efe3dbdc7026521da8576574c55120782f6cChris Banes        // smallest resolved dimension
1359840efe3dbdc7026521da8576574c55120782f6cChris Banes        final int d = Math.min(w, h);
1369840efe3dbdc7026521da8576574c55120782f6cChris Banes
1379840efe3dbdc7026521da8576574c55120782f6cChris Banes        // We add the shadow's padding to the measured dimension
1389840efe3dbdc7026521da8576574c55120782f6cChris Banes        setMeasuredDimension(
1399840efe3dbdc7026521da8576574c55120782f6cChris Banes                d + mShadowPadding.left + mShadowPadding.right,
1409840efe3dbdc7026521da8576574c55120782f6cChris Banes                d + mShadowPadding.top + mShadowPadding.bottom);
1419840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1429840efe3dbdc7026521da8576574c55120782f6cChris Banes
1439840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1449840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Set the ripple color for this {@link FloatingActionButton}.
1459840efe3dbdc7026521da8576574c55120782f6cChris Banes     * <p>
1469840efe3dbdc7026521da8576574c55120782f6cChris Banes     * When running on devices with KitKat or below, we draw a fill rather than a ripple.
1479840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1489840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param color ARGB color to use for the ripple.
1499840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1509840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setRippleColor(int color) {
1519840efe3dbdc7026521da8576574c55120782f6cChris Banes        if (mRippleColor != color) {
1529840efe3dbdc7026521da8576574c55120782f6cChris Banes            mRippleColor = color;
1539840efe3dbdc7026521da8576574c55120782f6cChris Banes            mImpl.setRippleColor(color);
1549840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
1559840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1569840efe3dbdc7026521da8576574c55120782f6cChris Banes
1579840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1589840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Return the tint applied to the background drawable, if specified.
1599840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1609840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @return the tint applied to the background drawable
1619840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @see #setBackgroundTintList(ColorStateList)
1629840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1639840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Nullable
1649840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
1659840efe3dbdc7026521da8576574c55120782f6cChris Banes    public ColorStateList getBackgroundTintList() {
1669840efe3dbdc7026521da8576574c55120782f6cChris Banes        return mBackgroundTint;
1679840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1689840efe3dbdc7026521da8576574c55120782f6cChris Banes
1699840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1709840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Applies a tint to the background drawable. Does not modify the current tint
1719840efe3dbdc7026521da8576574c55120782f6cChris Banes     * mode, which is {@link PorterDuff.Mode#SRC_IN} by default.
1729840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1739840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param tint the tint to apply, may be {@code null} to clear tint
1749840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1759840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setBackgroundTintList(@Nullable ColorStateList tint) {
1769840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setBackgroundTintList(tint);
1779840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1789840efe3dbdc7026521da8576574c55120782f6cChris Banes
1799840efe3dbdc7026521da8576574c55120782f6cChris Banes
1809840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1819840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Return the blending mode used to apply the tint to the background
1829840efe3dbdc7026521da8576574c55120782f6cChris Banes     * drawable, if specified.
1839840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1849840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @return the blending mode used to apply the tint to the background
1859840efe3dbdc7026521da8576574c55120782f6cChris Banes     *         drawable
1869840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @see #setBackgroundTintMode(PorterDuff.Mode)
1879840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
1889840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Nullable
1899840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
1909840efe3dbdc7026521da8576574c55120782f6cChris Banes    public PorterDuff.Mode getBackgroundTintMode() {
1919840efe3dbdc7026521da8576574c55120782f6cChris Banes        return mBackgroundTintMode;
1929840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
1939840efe3dbdc7026521da8576574c55120782f6cChris Banes
1949840efe3dbdc7026521da8576574c55120782f6cChris Banes    /**
1959840efe3dbdc7026521da8576574c55120782f6cChris Banes     * Specifies the blending mode used to apply the tint specified by
1969840efe3dbdc7026521da8576574c55120782f6cChris Banes     * {@link #setBackgroundTintList(ColorStateList)}} to the background
1979840efe3dbdc7026521da8576574c55120782f6cChris Banes     * drawable. The default mode is {@link PorterDuff.Mode#SRC_IN}.
1989840efe3dbdc7026521da8576574c55120782f6cChris Banes     *
1999840efe3dbdc7026521da8576574c55120782f6cChris Banes     * @param tintMode the blending mode used to apply the tint, may be
2009840efe3dbdc7026521da8576574c55120782f6cChris Banes     *                 {@code null} to clear tint
2019840efe3dbdc7026521da8576574c55120782f6cChris Banes     */
2029840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setBackgroundTintMode(@Nullable PorterDuff.Mode tintMode) {
2039840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setBackgroundTintMode(tintMode);
2049840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2059840efe3dbdc7026521da8576574c55120782f6cChris Banes
2069840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
2079840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void setBackgroundDrawable(Drawable background) {
2089840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.setBackgroundDrawable(background, mBackgroundTint, mBackgroundTintMode, mRippleColor);
2099840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2109840efe3dbdc7026521da8576574c55120782f6cChris Banes
2119840efe3dbdc7026521da8576574c55120782f6cChris Banes    final int getSizeDimension() {
2129840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (mSize) {
2139840efe3dbdc7026521da8576574c55120782f6cChris Banes            case SIZE_MINI:
2149840efe3dbdc7026521da8576574c55120782f6cChris Banes                return getResources().getDimensionPixelSize(R.dimen.fab_size_mini);
2159840efe3dbdc7026521da8576574c55120782f6cChris Banes            case SIZE_NORMAL:
2169840efe3dbdc7026521da8576574c55120782f6cChris Banes            default:
2179840efe3dbdc7026521da8576574c55120782f6cChris Banes                return getResources().getDimensionPixelSize(R.dimen.fab_size_normal);
2189840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
2199840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2209840efe3dbdc7026521da8576574c55120782f6cChris Banes
2219840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
2229840efe3dbdc7026521da8576574c55120782f6cChris Banes    protected void drawableStateChanged() {
2239840efe3dbdc7026521da8576574c55120782f6cChris Banes        super.drawableStateChanged();
2249840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.onDrawableStateChanged(getDrawableState());
2259840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2269840efe3dbdc7026521da8576574c55120782f6cChris Banes
2279840efe3dbdc7026521da8576574c55120782f6cChris Banes    @TargetApi(Build.VERSION_CODES.HONEYCOMB)
2289840efe3dbdc7026521da8576574c55120782f6cChris Banes    @Override
2299840efe3dbdc7026521da8576574c55120782f6cChris Banes    public void jumpDrawablesToCurrentState() {
2309840efe3dbdc7026521da8576574c55120782f6cChris Banes        super.jumpDrawablesToCurrentState();
2319840efe3dbdc7026521da8576574c55120782f6cChris Banes        mImpl.jumpDrawableToCurrentState();
2329840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2339840efe3dbdc7026521da8576574c55120782f6cChris Banes
2349840efe3dbdc7026521da8576574c55120782f6cChris Banes    private static int resolveAdjustedSize(int desiredSize, int measureSpec) {
2359840efe3dbdc7026521da8576574c55120782f6cChris Banes        int result = desiredSize;
2369840efe3dbdc7026521da8576574c55120782f6cChris Banes        int specMode = MeasureSpec.getMode(measureSpec);
2379840efe3dbdc7026521da8576574c55120782f6cChris Banes        int specSize = MeasureSpec.getSize(measureSpec);
2389840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (specMode) {
2399840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.UNSPECIFIED:
2409840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Parent says we can be as big as we want. Just don't be larger
2419840efe3dbdc7026521da8576574c55120782f6cChris Banes                // than max size imposed on ourselves.
2429840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = desiredSize;
2439840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
2449840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.AT_MOST:
2459840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Parent says we can be as big as we want, up to specSize.
2469840efe3dbdc7026521da8576574c55120782f6cChris Banes                // Don't be larger than specSize, and don't be larger than
2479840efe3dbdc7026521da8576574c55120782f6cChris Banes                // the max size imposed on ourselves.
2489840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = Math.min(desiredSize, specSize);
2499840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
2509840efe3dbdc7026521da8576574c55120782f6cChris Banes            case MeasureSpec.EXACTLY:
2519840efe3dbdc7026521da8576574c55120782f6cChris Banes                // No choice. Do what we are told.
2529840efe3dbdc7026521da8576574c55120782f6cChris Banes                result = specSize;
2539840efe3dbdc7026521da8576574c55120782f6cChris Banes                break;
2549840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
2559840efe3dbdc7026521da8576574c55120782f6cChris Banes        return result;
2569840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
2579840efe3dbdc7026521da8576574c55120782f6cChris Banes
2589840efe3dbdc7026521da8576574c55120782f6cChris Banes    static PorterDuff.Mode parseTintMode(int value, PorterDuff.Mode defaultMode) {
2599840efe3dbdc7026521da8576574c55120782f6cChris Banes        switch (value) {
2609840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 3:
2619840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_OVER;
2629840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 5:
2639840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_IN;
2649840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 9:
2659840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SRC_ATOP;
2669840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 14:
2679840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.MULTIPLY;
2689840efe3dbdc7026521da8576574c55120782f6cChris Banes            case 15:
2699840efe3dbdc7026521da8576574c55120782f6cChris Banes                return PorterDuff.Mode.SCREEN;
2709840efe3dbdc7026521da8576574c55120782f6cChris Banes            default:
2719840efe3dbdc7026521da8576574c55120782f6cChris Banes                return defaultMode;
2729840efe3dbdc7026521da8576574c55120782f6cChris Banes        }
2739840efe3dbdc7026521da8576574c55120782f6cChris Banes    }
27414d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes
27514d064edb3e4a16a3b90a4a850560177bea1e60dChris Banes    /**
276b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * Behavior designed for use with {@link FloatingActionButton} instances. It's main function
277b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * is to move {@link FloatingActionButton} views so that any displayed {@link Snackbar}s do
278b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     * not cover them.
279b7f9224b1495db47eb8fd813b5912250e900770aChris Banes     */
280b7f9224b1495db47eb8fd813b5912250e900770aChris Banes    public static class Behavior extends CoordinatorLayout.Behavior<FloatingActionButton> {
281b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        // We only support the FAB <> Snackbar shift movement on Honeycomb and above. This is
282b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        // because we can use view translation properties which greatly simplifies the code.
283b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        private static final boolean SNACKBAR_BEHAVIOR_ENABLED = Build.VERSION.SDK_INT >= 11;
284b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
285a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private Rect mTmpRect;
286a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private boolean mIsAnimatingOut;
287b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        private float mTranslationY;
288b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
289b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        @Override
290b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        public boolean layoutDependsOn(CoordinatorLayout parent,
291b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                FloatingActionButton child,
292b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                View dependency) {
293a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            // We're dependent on all SnackbarLayouts (if enabled)
294a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            return SNACKBAR_BEHAVIOR_ENABLED && dependency instanceof Snackbar.SnackbarLayout;
295b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
296b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
297b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        @Override
298b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        public boolean onDependentViewChanged(CoordinatorLayout parent, FloatingActionButton child,
299a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                View dependency) {
300a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            if (dependency instanceof Snackbar.SnackbarLayout) {
301a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                updateFabTranslationForSnackbar(parent, child, dependency);
302a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            } else if (dependency instanceof AppBarLayout) {
303a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                final AppBarLayout appBarLayout = (AppBarLayout) dependency;
304a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                if (mTmpRect == null) {
305a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    mTmpRect = new Rect();
306a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                }
307a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
308a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                // First, let's get the visible rect of the dependency
309a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                final Rect rect = mTmpRect;
310a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                ViewGroupUtils.getDescendantRect(parent, dependency, rect);
311a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
312a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                if (rect.bottom <= appBarLayout.getMinimumHeightForVisibleOverlappingContent()) {
313a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    // If the anchor's bottom is below the seam, we'll animate our FAB out
314a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    if (!mIsAnimatingOut && child.getVisibility() == View.VISIBLE) {
315a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        animateOut(child);
316a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    }
317a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                } else {
318a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    // Else, we'll animate our FAB back in
319a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    if (child.getVisibility() != View.VISIBLE) {
320a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        animateIn(child);
321a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    }
322a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                }
323a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            }
324b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            return false;
325b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
326b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
327a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private void updateFabTranslationForSnackbar(CoordinatorLayout parent,
328a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                FloatingActionButton fab, View snackbar) {
329a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            final float translationY = getFabTranslationYForSnackbar(parent, fab);
330b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            if (translationY != mTranslationY) {
331b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                // First, cancel any current animation
332b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                ViewCompat.animate(fab).cancel();
333b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
334b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                if (Math.abs(translationY - mTranslationY) == snackbar.getHeight()) {
335b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                    // If we're travelling by the height of the Snackbar then we probably need to
336b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                    // animate to the value
337a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    ViewCompat.animate(fab)
338a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            .translationY(translationY)
339a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            .setInterpolator(AnimationUtils.FAST_OUT_SLOW_IN_INTERPOLATOR)
340a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            .setListener(null);
341b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                } else {
342b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                    // Else we'll set use setTranslationY
343b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                    ViewCompat.setTranslationY(fab, translationY);
344b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                }
345b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                mTranslationY = translationY;
346b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            }
347b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
348b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
349a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private float getFabTranslationYForSnackbar(CoordinatorLayout parent,
350a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                FloatingActionButton fab) {
351b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            float minOffset = 0;
352a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            final List<View> dependencies = parent.getDependencies(fab);
353a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            for (int i = 0, z = dependencies.size(); i < z; i++) {
354a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                final View view = dependencies.get(i);
355a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                if (view instanceof Snackbar.SnackbarLayout && parent.doViewsOverlap(fab, view)) {
356a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    minOffset = Math.min(minOffset,
357a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            ViewCompat.getTranslationY(view) - view.getHeight());
358b7f9224b1495db47eb8fd813b5912250e900770aChris Banes                }
359b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            }
360a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
361b7f9224b1495db47eb8fd813b5912250e900770aChris Banes            return minOffset;
362b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
363b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
364a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private void animateIn(FloatingActionButton button) {
365a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            button.setVisibility(View.VISIBLE);
366a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
367a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            ViewCompat.animate(button)
368a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .scaleX(1f)
369a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .scaleY(1f)
370a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .alpha(1f)
371a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .setInterpolator(AnimationUtils.FAST_OUT_SLOW_IN_INTERPOLATOR)
372a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .withLayer()
373a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .setListener(null)
374a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .start();
375b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
376b7f9224b1495db47eb8fd813b5912250e900770aChris Banes
377a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes        private void animateOut(FloatingActionButton button) {
378a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes            ViewCompat.animate(button)
379a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .scaleX(0f)
380a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .scaleY(0f)
381a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .alpha(0f)
382a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .setInterpolator(AnimationUtils.FAST_OUT_SLOW_IN_INTERPOLATOR)
383a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .withLayer()
384a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    .setListener(new ViewPropertyAnimatorListener() {
385a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        @Override
386a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        public void onAnimationStart(View view) {
387a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            mIsAnimatingOut = true;
388a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        }
389a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
390a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        @Override
391a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        public void onAnimationCancel(View view) {
392a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            mIsAnimatingOut = false;
393a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        }
394a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes
395a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        @Override
396a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        public void onAnimationEnd(View view) {
397a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            mIsAnimatingOut = false;
398a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                            view.setVisibility(View.GONE);
399a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                        }
400a6a508b2296730ca6954aaebcca52a9962a5cb55Chris Banes                    }).start();
401b7f9224b1495db47eb8fd813b5912250e900770aChris Banes        }
402b7f9224b1495db47eb8fd813b5912250e900770aChris Banes    }
4039840efe3dbdc7026521da8576574c55120782f6cChris Banes}
404