我为我们在整个应用程序中广泛使用的控件编写了一个自定义小部件。 widget类派生自ImageButton
并以几种简单的方式扩展它。我已经定义了一个可以应用于小部件的样式,但是我更喜欢通过主题来设置它。在R.styleable
中,我看到了小工具样式属性,例如imageButtonStyle
和textViewStyle
。有没有办法为我写的自定义小部件创建类似的东西?
答案 0 :(得分:201)
是的,有一种方法:
假设您有一个小部件属性声明(在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
属性加载默认属性值。如果在主题中没有找到此类属性,或者属性的值为@null
,那么将使用obtainStyledAttributes
的最后一个参数:Widget.ImageButton.Custom
。
您可以更改所有实例和所有文件的名称(AndroidManifest.xml
除外),但最好使用Android命名约定。
答案 1 :(得分:4)
除了迈克尔的优秀答案之外,另一个方面是覆盖主题中的自定义属性。 假设您有许多自定义视图,这些视图都引用自定义属性&#34; custom_background&#34;。
<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>
你不能改变&#34; my_background_color&#34;的定义。在运行时,您可以轻松切换主题。