Відповіді:
Як правило, ви не можете змінити стилі програмно; ви можете встановити вигляд екрана, або частини макета, або окремої кнопки у вашому макеті XML, використовуючи теми чи стилі . Проте теми можна застосувати програмно .
Існує також така річ, StateListDrawable
що дозволяє вам визначати різні витяги для кожного стану, в якому ви Button
можете знаходитись, будь то зосереджений, вибраний, натиснутий, відключений тощо.
Наприклад, щоб змусити вашу кнопку змінити колір при натисканні, ви можете визначити XML-файл, який називається res/drawable/my_button.xml
каталог таким чином:
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item
android:state_pressed="true"
android:drawable="@drawable/btn_pressed" />
<item
android:state_pressed="false"
android:drawable="@drawable/btn_normal" />
</selector>
Потім ви можете застосувати цей селектор до а Button
, встановивши властивість android:background="@drawable/my_button"
.
style
атрибут Android , але ви можете програмно встановити тло, Button
як ви можете, з будь-яким іншим видом, якщо цього буде достатньо. Крім того, як Button
спадщина від TextView
, ви можете змінювати властивості тексту. Подивіться документацію API на ці елементи ... developer.android.com/reference/android/view/…
Перш за все, вам не потрібно використовувати надувний макет для створення простої кнопки. Ви можете просто використовувати:
button = new Button(context);
Якщо ви хочете стилізувати кнопку, у вас є два варіанти: найпростіший - просто вказати всі елементи в коді, як багато інших відповідей пропонують:
button.setTextColor(Color.RED);
button.setTextSize(TypedValue.COMPLEX_UNIT_SP, 18);
Інший варіант - визначити стиль у XML та застосувати його до кнопки. У загальному випадку ви можете використовувати ContextThemeWrapper
для цього:
ContextThemeWrapper newContext = new ContextThemeWrapper(baseContext, R.style.MyStyle);
button = new Button(newContext);
Для зміни текстових атрибутів TextView (або його підкласів типу Button) існує спеціальний метод:
button.setTextAppearance(context, R.style.MyTextStyle);
Цей останній не можна використовувати для зміни всіх атрибутів; наприклад, щоб змінити прокладки, ви повинні використовувати a ContextThemeWrapper
. Але для кольору тексту, розміру тощо ви можете використовувати setTextAppearance
.
Так, ви можете використовувати, наприклад, кнопку
Button b = new Button(this);
b.setBackgroundResource(R.drawable.selector_test);
Ви можете робити такі атрибути стилю:
Button myButton = new Button(this, null,android.R.attr.buttonBarButtonStyle);
замість:
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/btn"
style="?android:attr/buttonBarButtonStyle"
/>
Якщо ви використовуєте бібліотеку підтримки, ви можете просто її використовувати
TextViewCompat.setTextAppearance(textView, R.style.AppTheme_TextStyle_ButtonDefault_Whatever);
для TextViews та кнопок. Для решти переглядів є схожі класи :-)
Для всіх, хто шукає матеріальної відповіді, дивіться цю публікацію ТА: ТА Розмальовки кнопок в Android з Матеріал дизайну та AppCompat
Я використовував комбінацію цієї відповіді, щоб встановити колір тексту тексту за замовчуванням для білого кольору для моєї кнопки: https://stackoverflow.com/a/32238489/3075340
Потім відповідь https://stackoverflow.com/a/34355919/3075340 для програмного встановлення кольору фону. Код для цього:
ViewCompat.setBackgroundTintList(your_colored_button,
ContextCompat.getColorStateList(getContext(),R.color.your_custom_color));
your_colored_button
може бути просто звичайною Button
або кнопкою AppCompat, якщо ви хочете - я протестував вищевказаний код з обома типами кнопок, і він працює.
EDIT: Я виявив, що пристрої, що працюють перед льодяником, не працюють із наведеним вище кодом. Дивіться цю публікацію про те, як додати підтримку для пристроїв, що попередньо використовуються: https://stackoverflow.com/a/30277424/3075340
В основному це роблять:
Button b = (Button) findViewById(R.id.button);
ColorStateList c = ContextCompat.getColorStateList(mContext, R.color.your_custom_color;
Drawable d = b.getBackground();
if (b instanceof AppCompatButton) {
// appcompat button replaces tint of its drawable background
((AppCompatButton)b).setSupportBackgroundTintList(c);
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
// Lollipop button replaces tint of its drawable background
// however it is not equal to d.setTintList(c)
b.setBackgroundTintList(c);
} else {
// this should only happen if
// * manually creating a Button instead of AppCompatButton
// * LayoutInflater did not translate a Button to AppCompatButton
d = DrawableCompat.wrap(d);
DrawableCompat.setTintList(d, c);
b.setBackgroundDrawable(d);
}
Залежно від того, які атрибути стилю ви хочете змінити, ви можете використовувати паризьку бібліотеку:
Button view = (Button) LayoutInflater.from(this).inflate(R.layout.section_button, null);
Paris.style(view).apply(R.style.YourStyle);
Підтримуються багато атрибутів, таких як фон, набивання, розмір тексту, textColor тощо.
Відмова: Я був автором бібліотеки.
Відповідь @Dayerman та @h_rules правильна. Щоб надати детальний приклад з кодом, у папці, що малюється, створіть XML-файл під назвою button_disabled.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle" android:padding="10dp">
<solid android:color="@color/silver"/>
<corners
android:bottomRightRadius="20dp"
android:bottomLeftRadius="20dp"
android:topLeftRadius="20dp"
android:topRightRadius="20dp"/>
</shape>
Потім на Java,
((Button) findViewById(R.id.my_button)).setEnabled(false);
((Button) findViewById(R.id.my_button)).setBackgroundResource(R.drawable.button_disabled);
Це встановить властивість кнопки вимкнено, а колір встановить на сріблястий.
[Колір визначається в color.xml як:
<resources>
<color name="silver">#C0C0C0</color>
</resources>
Під час виконання роботи ви знаєте, у якому стилі ви хочете мати вашу кнопку. Тож заздалегідь у xml у папці макета ви можете мати всі готові до запуску кнопки із потрібними вам стилями. Отже, у папці макета у вас може бути файл з іменем: button_style_1.xml. Вміст цього файлу може виглядати так:
<?xml version="1.0" encoding="utf-8"?>
<Button
android:id="@+id/styleOneButton"
style="@style/FirstStyle" />
Якщо ви працюєте з фрагментами, то в onCreateView ви надуваєте цю кнопку, наприклад:
Button firstStyleBtn = (Button) inflater.inflate(R.layout.button_style_1, container, false);
де контейнер - контейнер ViewGroup, пов'язаний із методом onCreateView, який ви перекриваєте, створюючи фрагмент.
Потрібні ще дві такі кнопки? Ви створюєте їх так:
Button secondFirstStyleBtn = (Button) inflater.inflate(R.layout.button_style_1, container, false);
Button thirdFirstStyleBtn = (Button) inflater.inflate(R.layout.button_style_1, container, false);
Ви можете налаштувати ці кнопки:
secondFirstStyleBtn.setText("My Second");
thirdFirstStyleBtn.setText("My Third");
Потім ви додаєте свої спеціалізовані стилізовані кнопки до контейнера макета, який ви також надували методом onCreateView:
_stylizedButtonsContainer = (LinearLayout) rootView.findViewById(R.id.stylizedButtonsContainer);
_stylizedButtonsContainer.addView(firstStyleBtn);
_stylizedButtonsContainer.addView(secondFirstStyleBtn);
_stylizedButtonsContainer.addView(thirdFirstStyleBtn);
І саме так можна динамічно працювати зі стилізованими кнопками.
Я зробив для цього помічник інтерфейсу, використовуючи шаблон тримача.
public interface StyleHolder<V extends View> {
void applyStyle(V view);
}
Тепер для кожного стилю, який ви хочете використовувати прагматично, просто реалізуйте інтерфейс, наприклад:
public class ButtonStyleHolder implements StyleHolder<Button> {
private final Drawable background;
private final ColorStateList textColor;
private final int textSize;
public ButtonStyleHolder(Context context) {
TypedArray ta = context.obtainStyledAttributes(R.style.button, R.styleable.ButtonStyleHolder);
Resources resources = context.getResources();
background = ta.getDrawable(ta.getIndex(R.styleable.ButtonStyleHolder_android_background));
textColor = ta.getColorStateList(ta.getIndex(R.styleable.ButtonStyleHolder_android_textColor));
textSize = ta.getDimensionPixelSize(
ta.getIndex(R.styleable.ButtonStyleHolder_android_textSize),
resources.getDimensionPixelSize(R.dimen.standard_text_size)
);
// Don't forget to recycle!
ta.recycle();
}
@Override
public void applyStyle(Button btn) {
btn.setBackground(background);
btn.setTextColor(textColor);
btn.setTextSize(TypedValue.COMPLEX_UNIT_PX, textSize);
}
}
Заявіть про стилі в своєму attrs.xml
, стилі для цього прикладу є:
<declare-styleable name="ButtonStyleHolder">
<attr name="android:background" />
<attr name="android:textSize" />
<attr name="android:textColor" />
</declare-styleable>
Ось стиль, оголошений у styles.xml
:
<style name="button">
<item name="android:background">@drawable/button</item>
<item name="android:textColor">@color/light_text_color</item>
<item name="android:textSize">@dimen/standard_text_size</item>
</style>
І нарешті реалізація власника стилю:
Button btn = new Button(context);
StyleHolder<Button> styleHolder = new ButtonStyleHolder(context);
styleHolder.applyStyle(btn);
Я вважаю це дуже корисним, оскільки його можна легко використовувати повторно та зберігає код чистим та багатослівним, я б рекомендував використовувати це лише як локальну змінну, щоб ми могли дозволити колектору сміття виконувати свою роботу, коли ми закінчимо налаштування всіх стилів .
Я зіткнувся з тією ж проблемою нещодавно. ось як я це вирішив.
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<!-- This is the special two colors background START , after this LinearLayout, you can add all view that have it for main background-->
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:weightSum="2"
android:background="#FFFFFF"
android:orientation="horizontal"
>
<View
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_weight="1"
android:background="#0000FF" />
<View
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_weight="1"
android:background="#F000F0" />
</LinearLayout>
<!-- This is the special two colors background END-->
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:gravity="center"
android:text="This Text is centered with a special backgound,
You can add as much elements as you want as child of this RelativeLayout"
android:textColor="#FFFFFF"
android:textSize="20sp" />
</RelativeLayout>
Дякую !