Android: установить стиль просмотра программно

226

Вот XML:

<RelativeLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    style="@style/LightStyle"
    android:layout_width="fill_parent"
    android:layout_height="55dip"
    android:clickable="true"
    android:orientation="horizontal" >

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="horizontal" />

</RelativeLayout>

Как установить styleатрибут программно?

Джим
источник
Смотрите здесь ['ответ'] [1], это сработало для меня. [1]: stackoverflow.com/a/5488652/2938493
Artificioo

Ответы:

211

Технически вы можете применять стили программно, в любом случае с пользовательскими представлениями:

private MyRelativeLayout extends RelativeLayout {
  public MyRelativeLayout(Context context) {
     super(context, null, R.style.LightStyle);
  }
}

Конструктор с одним аргументом используется при создании экземпляров представлений программным способом.

Так что цепочка этого конструктора супер, который принимает параметр стиля.

RelativeLayout someLayout = new MyRelativeLayout(new ContextThemeWrapper(this,R.style.RadioButton));

Или, как @Dori указал просто:

RelativeLayout someLayout = new RelativeLayout(new ContextThemeWrapper(activity,R.style.LightStyle));
Бландел
источник
34
Вы можете просто сделать это программно, не расширяясь, так как конструктор 3 arg общедоступен в любом случае developer.android.com/reference/android/widget/… , android.util.AttributeSet, int)
Дори
1
@Turbo, оба документы и затмение должны сказать вам , что: http://developer.android.com/reference/android/widget/LinearLayout.html#LinearLayout(android.content.Context, android.util.AttributeSet, int). С LinearLayout требуется уровень 11+.
TheOne
2
@ Дори Что бы вы сдали за AttributeSet?
Бланделл
9
Если это TextView, вам нужно использовать setTextAppearance (R.style.small_text) для тех атрибутов, которые влияют на текст (размер, цвет и т. Д.)
Maragues
2
Я понятия не имею, почему не работает подход с тремя аргументами. Я обратился к Button. Подход @ Бенджамина Пиетта работает отлично.
Youngjae
124

Что сработало для меня:

Button b = new Button(new ContextThemeWrapper(this, R.style.ButtonText), null, 0);
  • Используйте ContextThemeWrapper

И

  • Используйте конструктор с тремя аргументами (без этого не получится)
Бенджамин Пиетт
источник
Использование вашего метода работает, но я получаю W / ResourceType oo Слишком много ссылок на атрибуты, остановлено на: 0x01010034. Любое решение?
HaloMediaz
Я никогда не видел эту проблему. Это может быть проблема конкретного устройства или что-то, связанное с последней версией Android?
Бенджамин Пиетт
11
ВАЖНЫЙ! Этот метод делает работу, но будет установить стиль для каждого зрения ребенка, а если создать новый макет с ContextThemeWrapper.
aProperFox,
@aProperFox вы можете объяснить лучше? Со мной работает, просто добавив кнопку. Вы ссылаетесь на макет?
Гилиан
1
@Gilian точно, если вы используете это в составном представлении, у которого есть один или несколько дочерних элементов, они получат тот же стиль, что и родительское представление, для которого вы установили стиль. Это может привести к слишком большому количеству отступов или отступов во внутренних представлениях и может привести к
безумию,
88

Обновление : во время ответа на этот вопрос (середина 2012 года, уровень API 14-15), программная настройка представления не была возможной (даже если были некоторые нетривиальные обходные пути), хотя это стало возможным после более позднего API релизы. Смотрите ответ @ Blundell для деталей.

СТАРЫЙ ответ:

Вы не можете установить стиль представления программно, но вы можете найти эту тему полезной.

Корхан Озтурк
источник
16
Не правда. Смотрите ответ @ Blundell.
a.bertucci
31
Для некоторых ситуаций (например, с использованием TextView) вы можете использовать textView.setTextAppearance(context, R.style.mystyle);. Согласно документу «Устанавливает цвет текста, размер, стиль, цвет подсказки и цвет выделения из указанного ресурса TextAppearance.» developer.android.com/reference/android/widget/… , int)
Рогель Гарсия,
4
api> 23; textView.setTextAppearance (R.style.mystyle);
alican akyol
1
Я отправил этот ответ более 4 лет назад @HaydenKai. с тех пор API сильно изменился, что позволило применять стили программно.
Корхан Озтюрк
3
@KorhanOzturk согласился, но на SO с такими вопросами, которые имеют активность, вопрос должен быть вновь открыт и принят новый ответ
HaydenKai
16

Для новой кнопки / TextView:

Button mMyButton = new Button(new ContextThemeWrapper(this, R.style.button_disabled), null, 0);

Для существующего экземпляра:

mMyButton.setTextAppearance(this, R.style.button_enabled);

Для изображения или макетов:

Image mMyImage = new ImageView(new ContextThemeWrapper(context, R.style.article_image), null, 0);
Алон Коган
источник
8

Если вы хотите продолжить использовать XML (что не позволяет сделать принятый ответ) и установить стиль после создания представления, вы можете использовать парижскую библиотеку, которая поддерживает подмножество всех доступных атрибутов.

Поскольку вы надуваете свое представление из XML, вам нужно указать идентификатор в макете:

<RelativeLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/my_styleable_relative_layout"
    style="@style/LightStyle"
    ...

Затем, когда вам нужно изменить стиль программно, после того, как макет был раздут:

// Any way to get the view instance will do
RelativeLayout myView = findViewById(R.id.my_styleable_relative_layout);

// This will apply all the supported attribute values of the style
Paris.style(myView).apply(R.style.LightStyle);

Более того: список поддерживаемых типов и атрибутов вида (включает фон, отступы, поля и т. Д. И может быть легко расширен) и инструкции по установке с дополнительной документацией .

Отказ от ответственности: я первый автор указанной библиотеки.

Нафанаил
источник
Это все еще рекомендуемый подход для использования в 2019 году?
Игорь Ганапольский
1
@IgorGanapolsky afaik да! Я не знаю ни одного лучшего.
Натанаэль
Спасибо @Nathanael за предоставление этого решения. Также для поддержки его с последним атрибутом line_height. Продолжайте хорошую работу! Действительно полезно.
Гиллем Рока
7

Вы можете применить стиль к своей деятельности, выполнив:

super.setTheme( R.style.MyAppTheme );

или Android по умолчанию:

super.setTheme( android.R.style.Theme );

в вашей деятельности, до setContentView().

FOMDeveloper
источник
4
@siemian, пожалуйста, позвольте FOM свободно выражать свое мнение? Он не нуждается в редактировании!
Мика,
6

Ни один из приведенных ответов не является правильным.

Вы можете установить стиль программно.

Краткий ответ: посмотрите на http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/5.1.1_r1/android/content/Context.java#435

Длинный ответ Вот мой фрагмент, чтобы программно установить пользовательский стиль на ваш взгляд:

1) Создайте стиль в вашем файле styles.xml

 <style name="MyStyle">
    <item name="customTextColor">#39445B</item>
    <item name="customDividerColor">#8D5AA8</item>
</style>

Не забудьте указать свои пользовательские атрибуты в файле attrs.xml

Мой файл attrsl.xml:

<declare-styleable name="CustomWidget">
    <attr name="customTextColor" format="color" />
    <attr name="customDividerColor" format="color" />
</declare-styleable>

Обратите внимание, что вы можете использовать любое имя для вашего стиля (мой CustomWidget)

Теперь давайте установим стиль для виджета программно. Вот мой простой виджет:

public class StyleableWidget extends LinearLayout {

private final StyleLoader styleLoader = new StyleLoader();

private TextView textView;
private View divider;

public StyleableWidget(Context context) {
    super(context);
    init();
}

private void init() {
    inflate(getContext(), R.layout.widget_styleable, this);
    textView = (TextView) findViewById(R.id.text_view);
    divider = findViewById(R.id.divider);
    setOrientation(VERTICAL);
}

protected void apply(StyleLoader.StyleAttrs styleAttrs) {
    textView.setTextColor(styleAttrs.textColor);
    divider.setBackgroundColor(styleAttrs.dividerColor);
}

public void setStyle(@StyleRes int style) {
    apply(styleLoader.load(getContext(), style));
}
}

расположение:

<TextView
    android:id="@+id/text_view"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textSize="22sp"
    android:layout_gravity="center"
    android:text="@string/styleble_title" />

<View
    android:id="@+id/divider"
    android:layout_width="match_parent"
    android:layout_height="1dp"/>

</merge>

И, наконец, реализация класса StyleLoader

public class StyleLoader {

public StyleLoader() {

}

public static class StyleAttrs {
    public int textColor;
    public int dividerColor;
}

public StyleAttrs load(Context context, @StyleRes int styleResId) {
    final TypedArray styledAttributes = context.obtainStyledAttributes(styleResId, R.styleable.CustomWidget);
    return load(styledAttributes);
}

@NonNull
private StyleAttrs load(TypedArray styledAttributes) {
    StyleAttrs styleAttrs = new StyleAttrs();
    try {
        styleAttrs.textColor = styledAttributes.getColor(R.styleable.CustomWidget_customTextColor, 0);
        styleAttrs.dividerColor = styledAttributes.getColor(R.styleable.CustomWidget_customDividerColor, 0);
    } finally {
        styledAttributes.recycle();
    }
    return styleAttrs;
}
}

Вы можете найти полностью рабочий пример на https://github.com/Defuera/SetStylableProgramatics

Defuera
источник
13
Это не настоящий стиль, это просто фальшивка, применяя / настраивая сохраненные настройки (да, на самом деле это какие-то настройки, сохраненные в вашем XML-файле) в ваших представлениях (здесь у вас есть только два фиксированных представления - и вам, безусловно, нужно знать их). заранее). Вся магия в applyметоде, который не делает ничего интересного. Истинное значение стиля заключается в том, что он автоматически применяет некоторый визуальный стиль ко всем будущим динамически добавляемым представлениям. Этот код здесь, конечно, не может этого сделать, и здесь нет ничего динамического, нам нужно знать, какие представления и какие свойства / поля устанавливать.
Король Король
1
Я не против указать, к каким представлениям применять стили, но в этом решении жестко заданы элементы стиля (например, textColor и dividerColor). Он не будет динамически находить все элементы, определенные в стиле, и применять их к представлению.
nasch
Ссылка, которую вы разместили, не открывается. Можете ли вы повторно опубликовать?
Игорь Ганапольский
4

Это довольно старый вопрос, но решение, которое сработало для меня сейчас, заключается в использовании 4-го параметра конструктора defStyleRes- если доступно .. на виду ... для установки стиля

Следующие работы для моих целей (kotlin):

val textView = TextView(context, null, 0, R.style.Headline1)
выпуклый корпус
источник
3

Это мой простой пример, ключом является ContextThemeWrapperобертка, без нее мой стиль не работает, и используется конструктор с тремя параметрами View.

ContextThemeWrapper themeContext = new ContextThemeWrapper(this, R.style.DefaultLabelStyle);
TextView tv = new TextView(themeContext, null, 0);
tv.setText("blah blah ...");
layout.addView(tv);
guogangj
источник
2

простой путь проходит через конструктор

RadioButton radioButton = new RadioButton(this,null,R.style.radiobutton_material_quiz);
Саи Гопи Н
источник
Еще немного иллюстрации о том, как использовать это решение, будет полезно.
Игорь Ганапольский
1

Я не предлагаю использовать ContextThemeWrapper, поскольку это делает это:

Указанная тема будет применена поверх темы базового контекста.

Что может привести к нежелательным результатам в вашем приложении. Вместо этого я предлагаю новую библиотеку «Париж» для этого от инженеров Airbnb:

https://github.com/airbnb/paris

Определите и примените стили к представлениям Android программно.

Но после некоторого времени использования я обнаружил, что он на самом деле довольно ограничен, и я перестал его использовать, потому что он не поддерживает много свойств, которые мне нужны, из коробки, поэтому нужно проверить и решить как всегда.

Renetik
источник
Объясните своего собеседника по снижению голосов ... Я потерял полтора дня, потому что я использовал ContextThemeWrapper, и он применился к моему белому фону контекстной темы, а затем внезапно
произошел
Разве парижская библиотека не использует ContextThemeWrapper ?
Игорь Ганапольский
@IgorGanapolsky это не так. Он читает стиль XML и преобразует его в соответствующие вызовы методов в представлении.
Натанаэль
Это на самом деле довольно ограничено, я перестал его использовать, потому что он не поддерживает много свойств, которые мне нужны ...
Renetik
-1

Я использовал представления, определенные в XML, в моей составной ViewGroup, надул их и добавил в Viewgroup. Таким образом, я не могу динамически менять стиль, но могу сделать некоторые настройки стиля. Мой композит:

public class CalendarView extends LinearLayout {

private GridView mCalendarGrid;
private LinearLayout mActiveCalendars;

private CalendarAdapter calendarAdapter;

public CalendarView(Context context) {
    super(context);

}

public CalendarView(Context context, AttributeSet attrs) {
    super(context, attrs);

}

@Override
protected void onFinishInflate() {
    super.onFinishInflate();
    init();
}

private void init() {
    mCalendarGrid = (GridView) findViewById(R.id.calendarContents);
    mCalendarGrid.setNumColumns(CalendarAdapter.NUM_COLS);

    calendarAdapter = new CalendarAdapter(getContext());
    mCalendarGrid.setAdapter(calendarAdapter);
    mActiveCalendars = (LinearLayout) findViewById(R.id.calendarFooter);
}

}

и мой взгляд в XML, где я могу назначить стили:

<com.mfitbs.android.calendar.CalendarView
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/calendar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:orientation="vertical"
>

<GridView
    android:id="@+id/calendarContents"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />

<LinearLayout
    android:id="@+id/calendarFooter"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal"
    />

user3918502
источник
1
Где вы динамически устанавливаете стиль?
Игорь Ганапольский
-3

Вы можете создать XML-файл, содержащий макет с желаемым стилем, а затем изменить фоновый ресурс вашего представления, например, так .

rfsbraz
источник
Вы говорите о «стиле», как в Android, но о «аспекте» кнопки. это очень разные.
Христос