Подтвердить что ты не робот

Как определить тему (стиль) для настраиваемого виджета

Я написал собственный виджет для элемента управления, который мы широко используем в нашем приложении. Класс виджета происходит от ImageButton и расширяет его несколькими путями. Я определил стиль, который я могу применить к виджету, как он использовался, но я бы предпочел установить это через тему. В R.styleable я вижу атрибуты стиля виджетов, такие как imageButtonStyle и textViewStyle. Есть ли способ создать что-то подобное для пользовательского виджета, который я написал?

4b9b3361

Ответ 1

Да, есть один способ:

Предположим, что у вас есть объявление атрибутов для вашего виджета (в 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.

Ответ 2

Другим аспектом в дополнение к превосходному ответу Майкла является переопределение пользовательских атрибутов в темы. Предположим, у вас есть несколько пользовательских представлений, которые все относятся к настраиваемому атрибуту "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:attr/colorBackground

Как большинство из вас заметили, вы можете - и, вероятно, должны использовать ссылки @color вместо жестко закодированных цветов.

Так почему бы просто не сделать

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

Вы не можете изменить определение "my_background_color" во время выполнения, тогда как вы можете легко переключать темы.