如何以编程方式更改CardView的背景颜色

时间:2014-11-07 18:14:19

标签: android android-5.0-lollipop android-cardview

Google有没有理由决定不采用动态更改CardView背景颜色的方法?

剪断here enter image description here


解决方法

简单的代码@Justin Powell建议对我不起作用。在Android 5.0上。但它确实让我朝着正确的方向前进。

此代码,(MyRoundRectDrawableWithShadow是this的副本)

        card.setBackgroundDrawable(new MyRoundRectDrawableWithShadow(context.getResources(),
                color,
                card.getRadius(),
                card.getCardElevation(),
                card.getMaxCardElevation()));

...给了我这个错误,

java.lang.NullPointerException: Attempt to invoke interface method 'void com.example.app.MyRoundRectDrawableWithShadow$RoundRectHelper.drawRoundRect(android.graphics.Canvas, android.graphics.RectF, float, android.graphics.Paint)' on a null object reference
        at com.example.app.MyRoundRectDrawableWithShadow.draw(MyRoundRectDrawableWithShadow.java:172)

其中只是说有一个被调用的接口是null。然后我查看了CardView source,了解它是如何做到的。我发现下面的一段代码以某种静态的方式初始化界面(我不明白为什么,如果你知道的话请解释),然后我在init初级调用一次,然后你可以设置卡片上面一大堆代码的颜色。

final RectF sCornerRect = new RectF();
MyRoundRectDrawableWithShadow.sRoundRectHelper
                = new MyRoundRectDrawableWithShadow.RoundRectHelper() {
            @Override
            public void drawRoundRect(Canvas canvas, RectF bounds, float cornerRadius,
                                      Paint paint) {
                final float twoRadius = cornerRadius * 2;
                final float innerWidth = bounds.width() - twoRadius;
                final float innerHeight = bounds.height() - twoRadius;
                sCornerRect.set(bounds.left, bounds.top,
                        bounds.left + cornerRadius * 2, bounds.top + cornerRadius * 2);
                canvas.drawArc(sCornerRect, 180, 90, true, paint);
                sCornerRect.offset(innerWidth, 0);
                canvas.drawArc(sCornerRect, 270, 90, true, paint);
                sCornerRect.offset(0, innerHeight);
                canvas.drawArc(sCornerRect, 0, 90, true, paint);
                sCornerRect.offset(-innerWidth, 0);
                canvas.drawArc(sCornerRect, 90, 90, true, paint);
                //draw top and bottom pieces
                canvas.drawRect(bounds.left + cornerRadius, bounds.top,
                        bounds.right - cornerRadius, bounds.top + cornerRadius,
                        paint);
                canvas.drawRect(bounds.left + cornerRadius,
                        bounds.bottom - cornerRadius, bounds.right - cornerRadius,
                        bounds.bottom, paint);
                //center
                canvas.drawRect(bounds.left, bounds.top + cornerRadius,
                        bounds.right, bounds.bottom - cornerRadius, paint);
            }
        };

然而,这个解决方案确实产生了一个新问题。不确定在棒棒糖前发生了什么,但是当CardView首次初始化时,它似乎创建了一个RoundRectDrawable作为您在XML中设置的属性的背景。当我们使用上面的代码更改颜色时,我们将MyRoundRectDrawableWithShadow设置为背景,如果您想再次更改颜色,则card.getRadius(),card.getCardElevation()等将不再起作用。

因此,首先尝试将它从CardView获取的背景解析为MyRoundRectDrawableWithShadow,然后从中获取值,如果成功的话(将在第二次+更改颜色时)。但是,如果它失败(这将是第一次换色,因为背景是一个不同的类),它将直接从CardView本身获取值。

    float cardRadius;
    float maxCardElevation;

    try{
        MyRoundRectDrawableWithShadow background = (MyRoundRectDrawableWithShadow)card.getBackground();
        cardRadius = background.getCornerRadius();
        maxCardElevation = background.getMaxShadowSize();
    }catch (ClassCastException classCastExeption){
        cardRadius = card.getRadius();
        maxCardElevation = card.getMaxCardElevation();
    }

    card.setBackgroundDrawable(
            new MyRoundRectDrawableWithShadow(context.getResources(),
                    Color.parseColor(note.getColor()),
                    cardRadius,
                    card.getCardElevation(),
                    maxCardElevation));

希望这是有道理的,我不是母语为英语的人......如上所述,这只是在Lollipop上测试的。

4 个答案:

答案 0 :(得分:18)

只是为了更新: 最新的支持库提供了直接功能:

CardView cardView;
cardView.setCardBackgroundColor(color);

答案 1 :(得分:4)

我不知道任何特定的推理。

然而,如果你对这个遗漏的黑客行为感兴趣......

使用此属性的所有CardView都使用颜色创建一个可绘制的圆角矩形,然后将其指定为CardView的背景。如果您确实想以编程方式设置颜色,可以创建RoundRectDrawableWithShadow的副本,然后执行以下操作:

mCardView.setBackgroundDrawable(new MyRoundRectDrawableWithShadow(getResources(), color, radius));

您无法继承 RoundRectDrawableWithShadow 或直接使用它,因为它不公开。

答案 2 :(得分:2)

这很棘手。您需要破解API才能完成此功能。 @Justin Powell's answer是正确的,但在API 21中崩溃。我的解决方案解决了这个问题。您需要添加这两个类:

MyRoundRectDrawableWithShadow

package android.support.v7.widget;

import android.content.res.Resources;

public class MyRoundRectDrawableWithShadow extends RoundRectDrawableWithShadow {

  public MyRoundRectDrawableWithShadow(Resources resources, int backgroundColor) {
    super(resources, backgroundColor,
        resources.getDimensionPixelSize(R.dimen.cardview_default_radius),
        resources.getDimensionPixelSize(R.dimen.cardview_default_elevation),
        resources.getDimensionPixelSize(R.dimen.cardview_default_elevation));
  }

  public MyRoundRectDrawableWithShadow(Resources resources, int backgroundColor, float radius) {
    super(resources, backgroundColor, radius,
        resources.getDimensionPixelSize(R.dimen.cardview_default_elevation),
        resources.getDimensionPixelSize(R.dimen.cardview_default_elevation));
  }

  public MyRoundRectDrawableWithShadow(Resources resources, int backgroundColor, float radius,
                                       float shadowSize, float maxShadowSize) {
    super(resources, backgroundColor, radius, shadowSize, maxShadowSize);
  }
}

MyRoundRectDrawable

package android.support.v7.widget;

import android.content.res.Resources;

public class MyRoundRectDrawable extends RoundRectDrawable {

  public MyRoundRectDrawable(Resources resources, int backgroundColor) {
    super(backgroundColor, resources.getDimensionPixelSize(R.dimen.cardview_default_radius));
  }

  public MyRoundRectDrawable(int backgroundColor, float radius) {
    super(backgroundColor, radius);
  }
}

然后使用此代码更改背景颜色:

final Drawable background;
if (Build.VERSION.SDK_INT >= 21) {
  background = new MyRoundRectDrawable(color);
} else {
  background = new MyRoundRectDrawableWithShadow(resources, color);
}
// This is to avoid to use a deprecated method
if (Build.VERSION.SDK_INT >= 16) {
  cardView.setBackground(background);
} else {
  cardView.setBackgroundDrawable(background);
}

答案 3 :(得分:0)

只需编写以下代码段: cardView.setCardBackgroundColor(getResources().getColor(R.color.colorPrimary)); 您想要代码的地方