添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

译自: Coloring Buttons w/ ThemeOverlays & Background Tints

如果我们想改变 Button 的背景色 background color ,一般如何实现?

本文将介绍两种实现方式。第一种是使用 AppCompat Widget.AppCompat.Button.Colored 样式并自定义 ThemeOverlay 来直接修改 Button 的背景色,而第二种方式则是使用 AppCompat built-in background tinting support 来实现同样的效果。

方式#1:通过 ThemeOverlay 方式修改

前面讲得有点空泛,在此之前我们先来了解下按钮的背景颜色是如何决定的。 material设计规范 对在 light and dark themes 中的按钮应该是什么样子有非常具体的要求。那么这些需求在底层是如何被满足的呢?

The Widget.AppCompat.Button button styles

为了回答上面所提出的问题,我们首先需要对 AppCompat 是如何决定一个标准按钮的默认外观有一个基本的理解, AppCompat 定义了大量的由 Widget.AppCompat.Button 派生出来的 style 样式,作为各种按钮的默认外观样式,在 Android 中,每一个 View 必须有一个默认的外观。也就使得 framework 可以为每一个控件应用一系列默认属性值,达到了更好的用户体验。对于按钮 Buttons ,默认的 Widget.AppCompat.Button 样式有以下特性:

  • 所有 Button 的最小宽高 minimum width minimum height 默认都是相同的(根据 material设计规范 分别指定为 88dp 48dp`)

  • 所有 Button TextAppearance 默认相同(例如:所有大写字母的文本默认字体和大小全都一致)

  • 所有 Button background 默认相同(例如:相同的颜色、圆角角度、相等的 insets padding 值等)

    Widget.AppCompat.Button 确保了所有 Button 默认情况下看起来大致相同。但是这些特性例如按钮背景色在 Light Dark 主题下又是如何被取决的,又或者是诸如 disabled , pressed , and focused 等状态下的情况呢?对于这些, AppCompat 主要取决于以下三种不同的属性( theme attributes ):

  • R.attr.colorButtonNormal :决定按钮在普通状态 normal state 下的背景色, light themes 主题下为 #ffd6d7d7 dark themes 主题下为 #ff5a595b

  • android.R.attr.disabledAlpha :浮点数类型,决定控件disabled 状态下的alpha值, light themes dark themes 主题下分别为 0.26f 0.30f

  • R.attr.colorControlHighlight :决定当控件被按下 pressed 或者获得焦点 focused 时,绘制在控件顶部的透明悬浮层的颜色 translucent overlay color (在 post-Lollipop 中的 Ripple 效果或者 pre-Lollipop list selectors 的前景色 foreground ),在 light themes dark themes 下分别表现为 12% black 20% white ( #1f000000 and #33ffffff )。

    AppCompat 在背后几乎已经为我们处理好了所有的事,它提供了另一个样式 Widget.AppCompat.Button.Colored 来 让改变按钮背景色 相对更加地简单,从它的命名可知,该样式派生于 Widget.AppCompat.Button ,因此拥有了所有和父类一样的属性,除了其中一个: R.attr.colorAccent 属性决定按钮的背景基色 button’s base background color

    Creating custom themes using ThemeOverlay s

    现在我们知道了按钮的背景可以通过 Widget.AppCompat.Button.Colored 样式来自定义,但是我们又该如何自定义 theme’s accent color 呢?一种是我们可以直接通过修改应用的主题中的 R.attr.colorAccent 属性来指定我们所要修改的颜色,但是大部分时间我们在应用中仅希望改变特定的几个按钮的背景,这个方法就变得不可取了。因为修改应用的全局Theme的属性会对应用中的所有按钮的背景有效。

    相反,一个更好的解决方案就是在 xml 中利用 android:theme ThemeOverlay 来为按钮指定自己的自定义主题。假设我们想改变 Google Red 500 的按钮背景色,为实现这一目标,我们可以定义以下主题:

    1
    2
    3
    4
    <!-- res/values/themes.xml -->
    <style name="RedButtonLightTheme" parent="ThemeOverlay.AppCompat.Light">
    <item name="colorAccent">@color/googred500</item>
    </style>

    然后设置到布局xml中对应的按钮上,如下:

    1
    2
    3
    4
    5
    <Button
    style="@style/Widget.AppCompat.Button.Colored"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/RedButtonLightTheme"/>

    就是这样!尽管达到了目的,我们仍想知道这个 ThemeOverlay 到底是什么。不像我们在 AndroidManifest.xml 文件中使用的主题(例如 Theme.AppCompat.Light , Theme.AppCompat.Dark 等), ThemeOverlay 仅定义了对控件外观设置Theme时最常用的一小部分 material-styled 的主题属性(参见 source code 获取完整的属性列表)。因此当我们仅想修改特定视图的一两个属性时就变得非常有用:只需要继承 ThemeOverlay ,然后更新我们想修改的属性的值,并且我们可以确保该视图除了已修改的属性外的其他属性使用的仍然是继承了的 light/dark 主题的值。

    方式#2: 设置 AppCompatButton background tint

    这里还有一种更加高效的方式来修改按钮的背景色,使用一个在 AppCompat 中叫做 background tinting 的新特性。我们可能知道许多框架的控件已经被替换为 AppCompat 样式,这使得 AppCompat 可以更好地通过 material design 规范来控制控件的着色 hint ,甚至是在 Lollipop 之前的设备。在应用运行期间, Buttons 变为 AppCompatButtons ImageViews 变为 AppCompatImageViews , CheckBoxs 变为 AppCompatCheckBoxs 等等等等。任何AppCompat化的控件只要实现了 TintableBackgroundView 接口,都可以运用 ColorStateList 属性来拥有自己的背景着色:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    <!-- res/color/btn_colored_background_tint.xml -->
    <selector xmlns:android="http://schemas.android.com/apk/res/android">

    <!-- Disabled state. -->
    <item android:state_enabled="false"
    android:color="?attr/colorButtonNormal"
    android:alpha="?android:attr/disabledAlpha"/>

    <!-- Enabled state. -->
    <item android:color="?attr/colorAccent"/>

    </selector>

    然后设置到布局 xml 中:

    1
    2
    3
    4
    <android.support.v7.widget.AppCompatButton
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    app:backgroundTint="@color/btn_colored_background_tint"/>

    或者指定通过方法 ViewCompat#setBackgroundTintList(View,ColorStateList) 动态指定:

    1
    2
    3
    final ColorStateList backgroundTintList =
    AppCompatResources.getColorStateList(context, R.color.btn_colored_background_tint);
    ViewCompat.setBackgroundTintList(button, backgroundTintList);

    就动态修改这种方式而言,尽管它更加有效( ThemeOverlays 只能定义在xml中而不能动态修改),但同时如果我们想确保按钮完全符合 material design 规范,则需要我们做更多的工作。我们可以创建一个简单的 BackgroundTints 工具类来让构造 colored background tint lists 更加快速简单:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    /**
    * Utility class for creating background tint {@link ColorStateList}s.
    */
    public final class BackgroundTints {
    private static final int[] DISABLED_STATE_SET = new int[]{-android.R.attr.state_enabled};
    private static final int[] PRESSED_STATE_SET = new int[]{android.R.attr.state_pressed};
    private static final int[] FOCUSED_STATE_SET = new int[]{android.R.attr.state_focused};
    private static final int[] EMPTY_STATE_SET = new int[0];

    /**
    * Returns a {@link ColorStateList} that can be used as a colored button's background tint.
    * Note that this code makes use of the {@code android.support.v4.graphics.ColorUtils}
    * utility class.
    */
    public static ColorStateList forColoredButton(Context context, @ColorInt int backgroundColor) {
    // On pre-Lollipop devices, we need 4 states total (disabled, pressed, focused, and default).
    // On post-Lollipop devices, we need 2 states total (disabled and default). The button's
    // RippleDrawable will animate the pressed and focused state changes for us automatically.
    final int numStates = Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP ? 4 : 2;

    final int[][] states = new int[numStates][];
    final int[] colors = new int[numStates];

    int i = 0;

    states[i] = DISABLED_STATE_SET;
    colors[i] = getDisabledButtonBackgroundColor(context);
    i++;

    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
    final int highlightedBackgroundColor = getHighlightedBackgroundColor(context, backgroundColor);

    states[i] = PRESSED_STATE_SET;
    colors[i] = highlightedBackgroundColor;
    i++;

    states[i] = FOCUSED_STATE_SET;
    colors[i] = highlightedBackgroundColor;
    i++;
    }

    states[i] = EMPTY_STATE_SET;
    colors[i] = backgroundColor;

    return new ColorStateList(states, colors);
    }

    /**
    * Returns the theme-dependent ARGB background color to use for disabled buttons.
    */
    @ColorInt
    private static int getDisabledButtonBackgroundColor(Context context) {
    // Extract the disabled alpha to apply to the button using the context's theme.
    // (0.26f for light themes and 0.30f for dark themes).
    final TypedValue tv = new TypedValue();
    context.getTheme().resolveAttribute(android.R.attr.disabledAlpha, tv, true);
    final float disabledAlpha = tv.getFloat();

    // Use the disabled alpha factor and the button's default normal color
    // to generate the button's disabled background color.
    final int colorButtonNormal = getThemeAttrColor(context, R.attr.colorButtonNormal);
    final int originalAlpha = Color.alpha(colorButtonNormal);
    return ColorUtils.setAlphaComponent(
    colorButtonNormal, Math.round(originalAlpha * disabledAlpha));
    }

    /**
    * Returns the theme-dependent ARGB color that results when colorControlHighlight is drawn
    * on top of the provided background color.
    */
    @ColorInt
    private static int getHighlightedBackgroundColor(Context context, @ColorInt int backgroundColor) {
    final int colorControlHighlight = getThemeAttrColor(context, R.attr.colorControlHighlight);
    return ColorUtils.compositeColors(colorControlHighlight, backgroundColor);
    }

    /** Returns the theme-dependent ARGB color associated with the provided theme attribute. */
    @ColorInt
    private static int getThemeAttrColor(Context context, @AttrRes int attr) {
    final TypedArray array = context.obtainStyledAttributes(null, new int[]{attr});
    try {
    return array.getColor(0, 0);
    } finally {
    array.recycle();
    }
    }

    private BackgroundTints() {}
    }

    通过它,我们可以简单地在代码中给按钮应用背景着色:

    1
    2
    ViewCompat.setBackgroundTintList(
    button, BackgroundTints.forColoredButton(button.getContext(), backgroundColor);

    测试(Pop quiz)!

    我们以一个简单的例子来检验一下前面所了解到的知识,在 App AndroidManifest.xml 中应用以下主题:

    1
    2
    3
    4
    5
    6
    <!-- res/values/themes.xml -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="colorPrimary">@color/indigo500</item>
    <item name="colorPrimaryDark">@color/indigo700</item>
    <item name="colorAccent">@color/pinkA200</item>
    </style>

    除此之外,再定义以下自定义主题:

    1
    2
    3
    4
    5
    6
    7
    8
    <!-- res/values/themes.xml -->
    <style name="RedButtonLightTheme" parent="ThemeOverlay.AppCompat.Light">
    <item name="colorAccent">@color/googred500</item>
    </style>

    <style name="RedButtonDarkTheme" parent="ThemeOverlay.AppCompat.Dark">
    <item name="colorAccent">@color/googred500</item>
    </style>

    下面 xml 中的按钮在默认、按下、关闭等状态时,在 API 19 and API 23 的设备会是什么样子的呢?

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"/>

    <Button
    style="@style/Widget.AppCompat.Button.Colored"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"/>

    <Button
    style="@style/Widget.AppCompat.Button.Colored"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/RedButtonLightTheme"/>

    <Button
    android:id="@+id/button4"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"/>

    <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/ThemeOverlay.AppCompat.Dark"/>

    <Button
    style="@style/Widget.AppCompat.Button.Colored"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/ThemeOverlay.AppCompat.Dark"/>

    <Button
    style="@style/Widget.AppCompat.Button.Colored"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/RedButtonDarkTheme"/>

    <Button
    android:id="@+id/button8"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:theme="@style/ThemeOverlay.AppCompat.Dark"/>

    </LinearLayout>

    假设第4和第8个按钮的背景着色是通过如下代码动态设置的:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    final int googRed500 = ContextCompat.getColor(activity, R.color.googred500);

    final View button4 = activity.findViewById(R.id.button4);
    ViewCompat.setBackgroundTintList(
    button4, BackgroundTints.forColoredButton(button4.getContext(), googRed500));

    final View button8 = activity.findViewById(R.id.button8);
    ViewCompat.setBackgroundTintList(
    button8, BackgroundTints.forColoredButton(button8.getContext(), googRed500));
  • API 19, default state
  • API 19, pressed state
  • API 19, disabled state
  • API 23, default state
  • API 23, pressed state
  • API 23, disabled state
  • (注意:截图中在disabled状态下的文字颜色错误的显示是一个已知的 Bug issue ,在 support library 的下一个版本更新中将得到修复)

    源码地址

    最后,感谢大家阅读。源码地址: source code for these examples on GitHub

    1. 方式#1:通过ThemeOverlay方式修改
    2. 方式#2: 设置AppCompatButton的background tint
    3. 源码地址