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

Здесь 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

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

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));

Ответ 2

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

Button b = new Button(new ContextThemeWrapper(this, R.style.ButtonText), null, 0);
  • Использование ContextThemeWrapper

и

  • Используйте конструктор 3-arguments (не будет работать без этого)

Ответ 3

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

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

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

Ответ 4

Для новой кнопки /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);

Ответ 5

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

super.setTheme( R.style.MyAppTheme );

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

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

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

Ответ 6

Если вы хотите продолжить использовать 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);

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

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

Ответ 7

Не указаны правильные ответы.

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

Короткий ответ - 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)

Теперь зададим стиль для виджета Programatically Вот мой простой виджет:

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/SetStylableProgramatically

Ответ 8

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

RadioButton radioButton = new RadioButton(this,null,R.style.radiobutton_material_quiz);

Ответ 9

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

ContextThemeWrapper themeContext = new ContextThemeWrapper(this, R.style.DefaultLabelStyle);
TextView tv = new TextView(themeContext, null, 0);
tv.setText("blah blah ...");
layout.addView(tv);

Ответ 10

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

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

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

https://github.com/airbnb/paris

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

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

Ответ 11

Я использовал представления, определенные в 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"
    />

Ответ 12

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