如何:为自定义窗口小部件定义主题(样式)项


86

我为控件编写了自定义窗口小部件,该控件在整个应用程序中广泛使用。小部件类ImageButton通过几种简单的方法派生并扩展它。我已经定义了一种样式,可以在使用时将其应用于小部件,但是我更愿意通过主题进行设置。在R.styleable我看到的小部件样式属性如imageButtonStyletextViewStyle。有什么方法可以为我编写的自定义窗口小部件创建类似的东西吗?

Answers:


209

是的,有一种方法:

假设您有一个小部件的属性声明(在中attrs.xml):

<declare-styleable name="CustomImageButton">
    <attr name="customAttr" format="string"/>
</declare-styleable>

声明用于样式参考的属性(在中attrs.xml):

<declare-styleable name="CustomTheme">
    <attr name="customImageButtonStyle" format="reference"/>
</declare-styleable>

为小部件声明一组默认属性值(在中styles.xml):

<style name="Widget.ImageButton.Custom" parent="android:style/Widget.ImageButton">
    <item name="customAttr">some value</item>
</style>

声明自定义主题(在中themes.xml):

<style name="Theme.Custom" parent="@android:style/Theme">
    <item name="customImageButtonStyle">@style/Widget.ImageButton.Custom</item>
</style>

将此属性用作窗口小部件的构造函数中的第三个参数(在中CustomImageButton.java):

public class CustomImageButton extends ImageButton {
    private String customAttr;

    public CustomImageButton( Context context ) {
        this( context, null );
    }

    public CustomImageButton( Context context, AttributeSet attrs ) {
        this( context, attrs, R.attr.customImageButtonStyle );
    }

    public CustomImageButton( Context context, AttributeSet attrs,
            int defStyle ) {
        super( context, attrs, defStyle );

        final TypedArray array = context.obtainStyledAttributes( attrs,
            R.styleable.CustomImageButton, defStyle,
            R.style.Widget_ImageButton_Custom ); // see below
        this.customAttr =
            array.getString( R.styleable.CustomImageButton_customAttr, "" );
        array.recycle();
    }
}

现在,您必须应用Theme.Custom所有使用的活动CustomImageButton(在AndroidManifest.xml中):

<activity android:name=".MyActivity" android:theme="@style/Theme.Custom"/>

就这样。现在CustomImageButton尝试从customImageButtonStyle当前主题的属性加载默认属性值。如果在主题或属性值中未找到此类属性@nullobtainStyledAttributes则将使用的最后一个参数:Widget.ImageButton.Custom在这种情况下。

您可以更改所有实例和所有文件的名称(除外AndroidManifest.xml),但是使用Android命名约定会更好。


4
有没有一种方法可以在不使用主题的情况下做完全相同的事情?我有一些自定义小部件,但我希望用户能够将其与所需的任何主题一起使用。按照您发布的方式进行操作需要用户使用我的主题。
theDazzler 2013年

3
@theDazzler:如果您的意思是开发人员可以使用的库,那么他们将能够创建自己的主题,并使用主题中的style属性(customImageButtonStyle在此答案中)指定小部件的样式。这就是在Android上自定义操作栏的方式。而且,如果您要在运行时更改主题,则使用这种方法是不可能的。
迈克尔,

@Michael,是的,我的意思是开发人员可以使用的库。您的评论解决了我的问题。谢谢!
theDazzler 2013年

30
所有这一切最令人惊讶的是,Google的某人认为这还可以。
格伦·梅纳德

1
是否name="CustomTheme"declare-styleable包装起任何作用的属性?那仅用于组织,因为我看不到它在任何地方使用。我可以将所有小部件的所有样式属性都放在一个包装中吗?
Tenfour04

4

除了michael的出色答案之外,另一方面是覆盖主题中的自定义属性。假设您有许多自定义视图,它们都引用自定义属性“ custom_background”。

<declare-styleable name="MyCustomStylables">
    <attr name="custom_background" format="color"/>
</declare-styleable>

在主题中定义值是什么

<style name="MyColorfulTheme" parent="AppTheme">
    <item name="custom_background">#ff0000</item>
</style>

要么

<style name="MyBoringTheme" parent="AppTheme">
    <item name="custom_background">#ffffff</item>
</style>

您可以使用样式来引用属性

<style name="MyDefaultLabelStyle" parent="AppTheme">
    <item name="android:background">?background_label</item>
</style>

注意问号,也用于参考android属性,如

?android:attr/colorBackground

正如大多数人所注意到的,您可以-并且可能应该-使用@color引用而不是硬编码的颜色。

那为什么不做

<item name="android:background">@color/my_background_color</item>

您不能在运行时更改“ my_background_color”的定义,而可以轻松切换主题。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.