Решение, обозначенное @ (Ted Hopp), работает, но требует небольшой поправки: в селекторе состояниям элементов требуется префикс «app:», иначе надувной модуль не распознает пространство имен правильно и не выполнит никаких действий; по крайней мере, это то, что происходит со мной.
Позвольте мне сообщить здесь все решение с некоторыми подробностями:
Сначала создайте файл res / values / attrs.xml:
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="food">
<attr name="state_fried" format="boolean" />
<attr name="state_baked" format="boolean" />
</declare-styleable>
</resources>
Затем определите свой собственный класс. Например, это может быть класс «FoodButton», производный от класса «Button». Вам нужно будет реализовать конструктор; реализовать этот, который, похоже, используется надувным устройством:
public FoodButton(Context context, AttributeSet attrs) {
super(context, attrs);
}
Вверху производного класса:
private static final int[] STATE_FRIED = {R.attr.state_fried};
private static final int[] STATE_BAKED = {R.attr.state_baked};
Также ваши переменные состояния:
private boolean mIsFried = false;
private boolean mIsBaked = false;
И пара сеттеров:
public void setFried(boolean isFried) {mIsFried = isFried;}
public void setBaked(boolean isBaked) {mIsBaked = isBaked;}
Затем переопределите функцию onCreateDrawableState:
@Override
protected int[] onCreateDrawableState(int extraSpace) {
final int[] drawableState = super.onCreateDrawableState(extraSpace + 2);
if (mIsFried) {
mergeDrawableStates(drawableState, STATE_FRIED);
}
if (mIsBaked) {
mergeDrawableStates(drawableState, STATE_BAKED);
}
return drawableState;
}
Наконец, самая тонкая часть этой головоломки; селектор, определяющий StateListDrawable, который вы будете использовать в качестве фона для своего виджета. Это файл "res / drawable / food_button.xml":
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res/com.mydomain.mypackage">
<item
app:state_baked="true"
app:state_fried="false"
android:drawable="@drawable/item_baked" />
<item
app:state_baked="false"
app:state_fried="true"
android:drawable="@drawable/item_fried" />
<item
app:state_baked="true"
app:state_fried="true"
android:drawable="@drawable/item_overcooked" />
<item
app:state_baked="false"
app:state_fried="false"
android:drawable="@drawable/item_raw" />
</selector>
Обратите внимание на префикс «app:», тогда как в стандартных состояниях Android вы бы использовали префикс «android:». Пространство имен XML имеет решающее значение для правильной интерпретации инфлятором и зависит от типа проекта, в который вы добавляете атрибуты. Если это приложение, замените com.mydomain.mypackage фактическим именем пакета вашего приложения (за исключением имени приложения). Если это библиотека, вы должны использовать «http://schemas.android.com/apk/res-auto» (и использовать Tools R17 или новее), иначе вы получите ошибки во время выполнения.
Пара замечаний:
Похоже, вам не нужно вызывать функцию «refreshDrawableState», по крайней мере, решение работает хорошо, в моем случае
Чтобы использовать свой собственный класс в XML-файле макета, вам нужно будет указать полное имя (например, com.mydomain.mypackage.FoodButton)
Вы можете смешивать стандартные состояния (например, android: нажат, android: включен, android: selected) с настраиваемыми состояниями, чтобы представить более сложные комбинации состояний.