34

我的波纹有以下代码:

<?xml version="1.0" encoding="utf-8"?>
<ripple xmlns:android="http://schemas.android.com/apk/res/android"
    android:color="?android:colorControlHighlight">
    <item android:id="@+id/rip">

        <shape android:shape="oval">
            <solid android:color="?android:colorAccent"/>
        </shape>
    </item>
</ripple>

现在我想让用户可以选择自己的颜色,所以我需要以编程方式创建波纹。
我发现了这个,我认为这是正确的方法,但我不知道如何处理。

波纹将在这里使用:

<ImageButton
    android:id="@+id/add_button"
    android:layout_width="@dimen/diameter"
    android:layout_height="@dimen/diameter"
    android:layout_gravity="end|bottom"
    android:layout_marginBottom="@dimen/add_button_margin"
    android:layout_marginEnd="@dimen/add_button_margin"
    android:layout_alignParentBottom="true"
    android:layout_alignParentEnd="true"
    android:src="@drawable/ic_action_add_person"
    android:tint="@android:color/white"
    android:background="@drawable/oval_ripple"
    android:elevation="@dimen/elevation_low"
    android:stateListAnimator="@anim/button_elevation"
    android:contentDescription="Neuer Spieler" />

我需要将背景设置为RippleDrawable这样的:

addButton.setBackground(ripple);
4

4 回答 4

66

这就是我能够做到这一点的方式。

请注意,这只是 Api 21+,因此如果您支持较低版本,则必须回退到正常的 Drawable。

public static RippleDrawable getPressedColorRippleDrawable(int normalColor, int pressedColor)
{
    return new RippleDrawable(getPressedColorSelector(normalColor, pressedColor), getColorDrawableFromColor(normalColor), null);
}

public static ColorStateList getPressedColorSelector(int normalColor, int pressedColor)
{
    return new ColorStateList(
        new int[][]
            {
                new int[]{android.R.attr.state_pressed},
                new int[]{android.R.attr.state_focused},
                new int[]{android.R.attr.state_activated},
                new int[]{}
            },
        new int[]
            {
                pressedColor,
                pressedColor,
                pressedColor,
                normalColor
            }
    );
}

public static ColorDrawable getColorDrawableFromColor(int color)
{
    return new ColorDrawable(color);
}

编辑: 我对此进行了更多修改,发现 ColorStateList 不需要像上述解决方案那样复杂。我已将其简化为以下代码段。(上面代码块中的其他所有内容都是相同的。我只更改了 ColorStateList 创建。)我将保留上面的块作为原始块,以防此方法不适用于某人。

new ColorStateList(
    new int[][]
        {
            new int[]{}
        },
    new int[]
        {
            pressedColor
        }
);
于 2015-02-05T20:59:29.050 回答
25
public static Drawable getAdaptiveRippleDrawable(
    int normalColor, int pressedColor) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        return new RippleDrawable(ColorStateList.valueOf(pressedColor),
                null, getRippleMask(normalColor));
    } else {
        return getStateListDrawable(normalColor, pressedColor);
    }
}

private static Drawable getRippleMask(int color) {
    float[] outerRadii = new float[8];
    // 3 is radius of final ripple, 
    // instead of 3 you can give required final radius
    Arrays.fill(outerRadii, 3);

    RoundRectShape r = new RoundRectShape(outerRadii, null, null);
    ShapeDrawable shapeDrawable = new ShapeDrawable(r);
    shapeDrawable.getPaint().setColor(color);
    return shapeDrawable;
}

public static StateListDrawable getStateListDrawable(
    int normalColor, int pressedColor) {
    StateListDrawable states = new StateListDrawable();
    states.addState(new int[]{android.R.attr.state_pressed}, 
        new ColorDrawable(pressedColor));
    states.addState(new int[]{android.R.attr.state_focused}, 
        new ColorDrawable(pressedColor));
    states.addState(new int[]{android.R.attr.state_activated}, 
        new ColorDrawable(pressedColor));
    states.addState(new int[]{}, 
        new ColorDrawable(normalColor));
    return states;
}

您可以获得可绘制对象并将其应用于任何视图view.setDrawable
对于 Lollipop+ 设备,您将获得波纹,否则它将改变视图的颜色。

于 2015-09-19T07:03:29.347 回答
2

基本上,您需要创建一个新的 RippleDrawable 对象。对于棒棒糖之前的设备,您需要一个 StateListDrawable(正如其他人已经建议的那样)。我用一堆与 Drawables 和着色相关的有用方法编写了一个有点足智多谋的 GIST: https ://gist.github.com/milosmns/6566ca9e3b756d922aa5

您很可能希望#getBackgroundDrawable()从该单例中使用。

于 2015-12-05T23:42:59.543 回答
0

您可以将其分配为前景,在 Kotlin 中:

view.foreground = ContextCompat.getDrawable(context, R.drawable.rippleRes)
于 2021-08-13T12:07:17.167 回答