Как: Определить тему (стиль) предмет для пользовательского виджета
-
12-10-2019 - |
Вопрос
Я написал пользовательский виджет для управления, который мы широко используем на протяжении всего нашего приложения. Класс виджетов происходит от ImageButton
и расширяет его несколькими простыми способами. Я определил стиль, который я могу применить к виджету по мере его использования, но я бы предпочел настроить его через тему. В R.styleable
Я вижу атрибуты стиля виджетов, такие как imageButtonStyle
а также textViewStyle
. Анкет Есть ли способ создать что -то подобное для пользовательского виджета, который я написал?
Решение
Да, есть один путь:
Предположим, у вас есть объявление атрибутов для вашего виджета (в 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.
Другие советы
Другим аспектом в дополнение к превосходному ответу Майкла является переоборудование пользовательских атрибутов в темах. Предположим, у вас есть несколько пользовательских представлений, которые все относятся к пользовательскому атрибуту «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» во время выполнения, тогда как вы можете легко переключать темы.