/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.support.v7.widget; import android.content.Context; import android.content.res.ColorStateList; import android.content.res.TypedArray; import android.graphics.Color; import android.graphics.Rect; import android.graphics.drawable.Drawable; import android.os.Build; import android.support.annotation.ColorInt; import android.support.annotation.Nullable; import android.support.v7.cardview.R; import android.util.AttributeSet; import android.view.View; import android.widget.FrameLayout; /** * A FrameLayout with a rounded corner background and shadow. *
* CardView uses elevation
property on Lollipop for shadows and falls back to a
* custom emulated shadow implementation on older platforms.
*
* Due to expensive nature of rounded corner clipping, on platforms before Lollipop, CardView does * not clip its children that intersect with rounded corners. Instead, it adds padding to avoid such * intersection (See {@link #setPreventCornerOverlap(boolean)} to change this behavior). *
* Before Lollipop, CardView adds padding to its content and draws shadows to that area. This
* padding amount is equal to maxCardElevation + (1 - cos45) * cornerRadius
on the
* sides and maxCardElevation * 1.5 + (1 - cos45) * cornerRadius
on top and bottom.
*
* Since padding is used to offset content for shadows, you cannot set padding on CardView. * Instead, you can use content padding attributes in XML or * {@link #setContentPadding(int, int, int, int)} in code to set the padding between the edges of * the CardView and children of CardView. *
* Note that, if you specify exact dimensions for the CardView, because of the shadows, its content
* area will be different between platforms before Lollipop and after Lollipop. By using api version
* specific resource values, you can avoid these changes. Alternatively, If you want CardView to add
* inner padding on platforms Lollipop and after as well, you can call
* {@link #setUseCompatPadding(boolean)} and pass true
.
*
* To change CardView's elevation in a backward compatible way, use
* {@link #setCardElevation(float)}. CardView will use elevation API on Lollipop and before
* Lollipop, it will change the shadow size. To avoid moving the View while shadow size is changing,
* shadow size is clamped by {@link #getMaxCardElevation()}. If you want to change elevation
* dynamically, you should call {@link #setMaxCardElevation(float)} when CardView is initialized.
*
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardBackgroundColor
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardCornerRadius
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardElevation
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardMaxElevation
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardUseCompatPadding
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardPreventCornerOverlap
* @attr ref android.support.v7.cardview.R.styleable#CardView_contentPadding
* @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingLeft
* @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingTop
* @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingRight
* @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingBottom
*/
public class CardView extends FrameLayout {
private static final int[] COLOR_BACKGROUND_ATTR = {android.R.attr.colorBackground};
private static final CardViewImpl IMPL;
static {
if (Build.VERSION.SDK_INT >= 21) {
IMPL = new CardViewApi21Impl();
} else if (Build.VERSION.SDK_INT >= 17) {
IMPL = new CardViewApi17Impl();
} else {
IMPL = new CardViewBaseImpl();
}
IMPL.initStatic();
}
private boolean mCompatPadding;
private boolean mPreventCornerOverlap;
/**
* CardView requires to have a particular minimum size to draw shadows before API 21. If
* developer also sets min width/height, they might be overridden.
*
* CardView works around this issue by recording user given parameters and using an internal
* method to set them.
*/
int mUserSetMinWidth, mUserSetMinHeight;
final Rect mContentPadding = new Rect();
final Rect mShadowBounds = new Rect();
public CardView(Context context) {
super(context);
initialize(context, null, 0);
}
public CardView(Context context, AttributeSet attrs) {
super(context, attrs);
initialize(context, attrs, 0);
}
public CardView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
initialize(context, attrs, defStyleAttr);
}
@Override
public void setPadding(int left, int top, int right, int bottom) {
// NO OP
}
@Override
public void setPaddingRelative(int start, int top, int end, int bottom) {
// NO OP
}
/**
* Returns whether CardView will add inner padding on platforms Lollipop and after.
*
* @return true
if CardView adds inner padding on platforms Lollipop and after to
* have same dimensions with platforms before Lollipop.
*/
public boolean getUseCompatPadding() {
return mCompatPadding;
}
/**
* CardView adds additional padding to draw shadows on platforms before Lollipop.
*
* This may cause Cards to have different sizes between Lollipop and before Lollipop. If you
* need to align CardView with other Views, you may need api version specific dimension
* resources to account for the changes.
* As an alternative, you can set this flag to true
and CardView will add the same
* padding values on platforms Lollipop and after.
*
* Since setting this flag to true adds unnecessary gaps in the UI, default value is
* false
.
*
* @param useCompatPadding true>
if CardView should add padding for the shadows on
* platforms Lollipop and above.
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardUseCompatPadding
*/
public void setUseCompatPadding(boolean useCompatPadding) {
if (mCompatPadding != useCompatPadding) {
mCompatPadding = useCompatPadding;
IMPL.onCompatPaddingChanged(mCardViewDelegate);
}
}
/**
* Sets the padding between the Card's edges and the children of CardView.
*
* Depending on platform version or {@link #getUseCompatPadding()} settings, CardView may * update these values before calling {@link android.view.View#setPadding(int, int, int, int)}. * * @param left The left padding in pixels * @param top The top padding in pixels * @param right The right padding in pixels * @param bottom The bottom padding in pixels * @attr ref android.support.v7.cardview.R.styleable#CardView_contentPadding * @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingLeft * @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingTop * @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingRight * @attr ref android.support.v7.cardview.R.styleable#CardView_contentPaddingBottom */ public void setContentPadding(int left, int top, int right, int bottom) { mContentPadding.set(left, top, right, bottom); IMPL.updatePadding(mCardViewDelegate); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { if (!(IMPL instanceof CardViewApi21Impl)) { final int widthMode = MeasureSpec.getMode(widthMeasureSpec); switch (widthMode) { case MeasureSpec.EXACTLY: case MeasureSpec.AT_MOST: final int minWidth = (int) Math.ceil(IMPL.getMinWidth(mCardViewDelegate)); widthMeasureSpec = MeasureSpec.makeMeasureSpec(Math.max(minWidth, MeasureSpec.getSize(widthMeasureSpec)), widthMode); break; } final int heightMode = MeasureSpec.getMode(heightMeasureSpec); switch (heightMode) { case MeasureSpec.EXACTLY: case MeasureSpec.AT_MOST: final int minHeight = (int) Math.ceil(IMPL.getMinHeight(mCardViewDelegate)); heightMeasureSpec = MeasureSpec.makeMeasureSpec(Math.max(minHeight, MeasureSpec.getSize(heightMeasureSpec)), heightMode); break; } super.onMeasure(widthMeasureSpec, heightMeasureSpec); } else { super.onMeasure(widthMeasureSpec, heightMeasureSpec); } } private void initialize(Context context, AttributeSet attrs, int defStyleAttr) { TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CardView, defStyleAttr, R.style.CardView); ColorStateList backgroundColor; if (a.hasValue(R.styleable.CardView_cardBackgroundColor)) { backgroundColor = a.getColorStateList(R.styleable.CardView_cardBackgroundColor); } else { // There isn't one set, so we'll compute one based on the theme final TypedArray aa = getContext().obtainStyledAttributes(COLOR_BACKGROUND_ATTR); final int themeColorBackground = aa.getColor(0, 0); aa.recycle(); // If the theme colorBackground is light, use our own light color, otherwise dark final float[] hsv = new float[3]; Color.colorToHSV(themeColorBackground, hsv); backgroundColor = ColorStateList.valueOf(hsv[2] > 0.5f ? getResources().getColor(R.color.cardview_light_background) : getResources().getColor(R.color.cardview_dark_background)); } float radius = a.getDimension(R.styleable.CardView_cardCornerRadius, 0); float elevation = a.getDimension(R.styleable.CardView_cardElevation, 0); float maxElevation = a.getDimension(R.styleable.CardView_cardMaxElevation, 0); mCompatPadding = a.getBoolean(R.styleable.CardView_cardUseCompatPadding, false); mPreventCornerOverlap = a.getBoolean(R.styleable.CardView_cardPreventCornerOverlap, true); int defaultPadding = a.getDimensionPixelSize(R.styleable.CardView_contentPadding, 0); mContentPadding.left = a.getDimensionPixelSize(R.styleable.CardView_contentPaddingLeft, defaultPadding); mContentPadding.top = a.getDimensionPixelSize(R.styleable.CardView_contentPaddingTop, defaultPadding); mContentPadding.right = a.getDimensionPixelSize(R.styleable.CardView_contentPaddingRight, defaultPadding); mContentPadding.bottom = a.getDimensionPixelSize(R.styleable.CardView_contentPaddingBottom, defaultPadding); if (elevation > maxElevation) { maxElevation = elevation; } mUserSetMinWidth = a.getDimensionPixelSize(R.styleable.CardView_android_minWidth, 0); mUserSetMinHeight = a.getDimensionPixelSize(R.styleable.CardView_android_minHeight, 0); a.recycle(); IMPL.initialize(mCardViewDelegate, context, backgroundColor, radius, elevation, maxElevation); } @Override public void setMinimumWidth(int minWidth) { mUserSetMinWidth = minWidth; super.setMinimumWidth(minWidth); } @Override public void setMinimumHeight(int minHeight) { mUserSetMinHeight = minHeight; super.setMinimumHeight(minHeight); } /** * Updates the background color of the CardView * * @param color The new color to set for the card background * @attr ref android.support.v7.cardview.R.styleable#CardView_cardBackgroundColor */ public void setCardBackgroundColor(@ColorInt int color) { IMPL.setBackgroundColor(mCardViewDelegate, ColorStateList.valueOf(color)); } /** * Updates the background ColorStateList of the CardView * * @param color The new ColorStateList to set for the card background * @attr ref android.support.v7.cardview.R.styleable#CardView_cardBackgroundColor */ public void setCardBackgroundColor(@Nullable ColorStateList color) { IMPL.setBackgroundColor(mCardViewDelegate, color); } /** * Returns the background color state list of the CardView. * * @return The background color state list of the CardView. */ public ColorStateList getCardBackgroundColor() { return IMPL.getBackgroundColor(mCardViewDelegate); } /** * Returns the inner padding after the Card's left edge * * @return the inner padding after the Card's left edge */ public int getContentPaddingLeft() { return mContentPadding.left; } /** * Returns the inner padding before the Card's right edge * * @return the inner padding before the Card's right edge */ public int getContentPaddingRight() { return mContentPadding.right; } /** * Returns the inner padding after the Card's top edge * * @return the inner padding after the Card's top edge */ public int getContentPaddingTop() { return mContentPadding.top; } /** * Returns the inner padding before the Card's bottom edge * * @return the inner padding before the Card's bottom edge */ public int getContentPaddingBottom() { return mContentPadding.bottom; } /** * Updates the corner radius of the CardView. * * @param radius The radius in pixels of the corners of the rectangle shape * @attr ref android.support.v7.cardview.R.styleable#CardView_cardCornerRadius * @see #setRadius(float) */ public void setRadius(float radius) { IMPL.setRadius(mCardViewDelegate, radius); } /** * Returns the corner radius of the CardView. * * @return Corner radius of the CardView * @see #getRadius() */ public float getRadius() { return IMPL.getRadius(mCardViewDelegate); } /** * Updates the backward compatible elevation of the CardView. * * @param elevation The backward compatible elevation in pixels. * @attr ref android.support.v7.cardview.R.styleable#CardView_cardElevation * @see #getCardElevation() * @see #setMaxCardElevation(float) */ public void setCardElevation(float elevation) { IMPL.setElevation(mCardViewDelegate, elevation); } /** * Returns the backward compatible elevation of the CardView. * * @return Elevation of the CardView * @see #setCardElevation(float) * @see #getMaxCardElevation() */ public float getCardElevation() { return IMPL.getElevation(mCardViewDelegate); } /** * Updates the backward compatible maximum elevation of the CardView. *
* Calling this method has no effect if device OS version is Lollipop or newer and
* {@link #getUseCompatPadding()} is false
.
*
* @param maxElevation The backward compatible maximum elevation in pixels.
* @attr ref android.support.v7.cardview.R.styleable#CardView_cardMaxElevation
* @see #setCardElevation(float)
* @see #getMaxCardElevation()
*/
public void setMaxCardElevation(float maxElevation) {
IMPL.setMaxElevation(mCardViewDelegate, maxElevation);
}
/**
* Returns the backward compatible maximum elevation of the CardView.
*
* @return Maximum elevation of the CardView
* @see #setMaxCardElevation(float)
* @see #getCardElevation()
*/
public float getMaxCardElevation() {
return IMPL.getMaxElevation(mCardViewDelegate);
}
/**
* Returns whether CardView should add extra padding to content to avoid overlaps with rounded
* corners on pre-Lollipop platforms.
*
* @return True if CardView prevents overlaps with rounded corners on platforms before Lollipop.
* Default value is true
.
*/
public boolean getPreventCornerOverlap() {
return mPreventCornerOverlap;
}
/**
* On pre-Lollipop platforms, CardView does not clip the bounds of the Card for the rounded
* corners. Instead, it adds padding to content so that it won't overlap with the rounded
* corners. You can disable this behavior by setting this field to false
.
*
* Setting this value on Lollipop and above does not have any effect unless you have enabled * compatibility padding. * * @param preventCornerOverlap Whether CardView should add extra padding to content to avoid * overlaps with the CardView corners. * @attr ref android.support.v7.cardview.R.styleable#CardView_cardPreventCornerOverlap * @see #setUseCompatPadding(boolean) */ public void setPreventCornerOverlap(boolean preventCornerOverlap) { if (preventCornerOverlap != mPreventCornerOverlap) { mPreventCornerOverlap = preventCornerOverlap; IMPL.onPreventCornerOverlapChanged(mCardViewDelegate); } } private final CardViewDelegate mCardViewDelegate = new CardViewDelegate() { private Drawable mCardBackground; @Override public void setCardBackground(Drawable drawable) { mCardBackground = drawable; setBackgroundDrawable(drawable); } @Override public boolean getUseCompatPadding() { return CardView.this.getUseCompatPadding(); } @Override public boolean getPreventCornerOverlap() { return CardView.this.getPreventCornerOverlap(); } @Override public void setShadowPadding(int left, int top, int right, int bottom) { mShadowBounds.set(left, top, right, bottom); CardView.super.setPadding(left + mContentPadding.left, top + mContentPadding.top, right + mContentPadding.right, bottom + mContentPadding.bottom); } @Override public void setMinWidthHeightInternal(int width, int height) { if (width > mUserSetMinWidth) { CardView.super.setMinimumWidth(width); } if (height > mUserSetMinHeight) { CardView.super.setMinimumHeight(height); } } @Override public Drawable getCardBackground() { return mCardBackground; } @Override public View getCardView() { return CardView.this; } }; }