Попытка получить значения атрибута в коде возвращает неверные значения
Я хочу извлечь несколько атрибутов из ресурса стиля (интересуются только атрибутами, входящими в группу TextAppearance)
Стиль, определяемый как
<style name="Label" parent="@android:style/TextAppearance.Small">
<item name="android:textColor">@color/floatlabel_text</item>
<item name="android:textSize">8dp</item>
<item name="android:textStyle">bold</item>
</style>
Сначала попробуйте
Сначала я попытался использовать TextView (строки 663-731), но потом выяснилось, что у нас нет доступа к com.android.internal.R
Частичное решение
Вот почему я переключился на это решение: qaru.site/info/77255/...
Итак, я создал textAppearanceAttr для замены com.android.internal.R.styleable.TextAppearance(содержит только 10/13 атрибутов TextAppearance, которые меня интересуют)
int[] textAppearanceAttr = new int[]{
android.R.attr.textColor,
android.R.attr.textSize,
android.R.attr.typeface,
android.R.attr.fontFamily,
android.R.attr.textStyle,
android.R.attr.textAllCaps,
android.R.attr.shadowColor,
android.R.attr.shadowDx,
android.R.attr.shadowDy,
android.R.attr.shadowRadius};
Вот как я его использовал. Я получаю идентификатор ресурса стиля (ресурс ссылается на атрибут clTextAppearance)
int ap = a.getResourceId(R.styleable.CustomLabelLayout_clTextAppearance, android.R.style.TextAppearance_Small);
TypedArray appearance = mContext.obtainStyledAttributes(ap, textAppearanceAttr);
И вот как я получаю атрибуты (все еще следуя ответам по ссылке выше):
mLabelTextColor = appearance.getColorStateList(0);
mLabelTextSize = appearance.getDimensionPixelSize(1, 15);
mLabelTypeface = appearance.getInt(2, -1);
mLabelFontFamily = appearance.getString(3);
mLabelTextStyle = appearance.getInt(4, -1);
(5 more...)
Текущий выпуск
Кажется, что установлен только атрибут first, каждый другой либо устанавливает значение по умолчанию, либо значение null.
Взлом, который, кажется, работает
Индивидуальные массивы:
int[] textSizeAttr = new int[] { android.R.attr.textSize};
int[] textStyleAttr = new int[] { android.R.attr.textStyle};
И получите такие атрибуты
appearance.recycle();
appearance = mContext.obtainStyledAttributes(ap, textSizeAttr);
mLabelTextSize = appearance.getDimensionPixelSize(0, 15);
appearance.recycle();
appearance = mContext.obtainStyledAttributes(ap, textStyleAttr);
mLabelTextStyle = appearance.getInt(0, -1);
appearance.recycle();
Теперь это такая трата.
Вопросы
- Я хотел бы знать, почему получение всех атрибутов сразу не работает.
- Есть ли решение (где вся дополнительная работа не нужна)?
РЕДАКТИРОВАТЬ 1
Я нашел здесь нечто похожее: qaru.site/info/95987/...
И по какой-то причине это работает. Пока я не добавлю больше атрибутов в массив, все становится кером.
Пример:
int[] attrs = {android.R.attr.textColor,
android.R.attr.textSize,
android.R.attr.background,
android.R.attr.textStyle,
android.R.attr.textAppearance,
android.R.attr.textColorLink,
android.R.attr.orientation,
android.R.attr.text};
Если я получаю текст с использованием указанного массива, он работает.
String text = ta.getString(7);
Но если я изменил массив на ниже, он не сработал (заменил android.R.attr.orientation на android.R.attr.shadowColor)
int[] attrs = {android.R.attr.textColor,
android.R.attr.textSize,
android.R.attr.background,
android.R.attr.textStyle,
android.R.attr.textAppearance,
android.R.attr.textColorLink,
android.R.attr.shadowColor,
android.R.attr.text};
Почему это происходит? (Вопрос № 1)
Ответы
Ответ 1
Я думаю, у меня есть идея, почему это происходит. Похоже, если идентификаторы не отсортированы, возникают проблемы. textColor
, например, имеет самое низкое значение int
, поэтому он начинает работать, помещаясь в первую позицию в массиве.
Если вы посмотрите на R.java
на свой стиль, вы увидите, что компилятор ресурсов Android отсортировал идентификаторы для вас. Поэтому он всегда работает, если вы объявляете stylable в attrs.xml
и не можете работать, если вручную создаете массивы идентификаторов.
Я считаю, что для сортировки идентификаторов есть причина производительности. Если они отсортированы, атрибуты могут быть прочитаны из AttributeSet
, используя один обход вместо N обходов в случае N идентификаторов.
UPDATE:
Я взглянул на исходный код, и это доказывает мою идею.
Context.obtainStyledAttributes() вызывает метод JNI AssetManager.applyStyle().
Вы можете найти источник здесь:
https://android.googlesource.com/platform/frameworks/base.git/+/android-4.3_r2.1/core/jni/android_util_AssetManager.cpp
В строке 1001 вы найдете цикл while, где ix (индекс в извлеченном массиве атрибутов XML) всегда увеличивается и никогда reset равно 0. Это означает, что если textColor является последним индексом в массиве (переменная "src" в коде), то мы никогда не доберемся до этого атрибута.
Ответ 2
Благодаря @PrivatMamtora и @igret для изучения этого!
Если проблема в том, что идентификаторы должны быть заказаны, это должно быть нормально.
private static final int ATTR_PADDING = android.R.attr.padding;
private static final int ATTR_TEXT_COLOR = android.R.attr.textColor;
private static final int ATTR_TEXT_SIZE = android.R.attr.textSize;
private void loadAttributes(Context context, AttributeSet attrs) {
int[] ids = { ATTR_PADDING, ATTR_TEXT_COLOR, ATTR_TEXT_SIZE};
Arrays.sort(ids); // just sort the array
TypedArray a = context.obtainStyledAttributes(attrs, ids);
try {
padding = a.getDimensionPixelSize(indexOf(ATTR_PADDING, ids), padding);
textColor = a.getColor(indexOf(ATTR_TEXT_COLOR, ids), textColor);
textSize = a.getDimensionPixelSize(indexOf(ATTR_TEXT_SIZE, ids), textSize);
} finally {
a.recycle();
}
}
private int indexOf(int id, int[] ids) {
for (int i = 0; i < ids.length; i++) {
if (ids[i] == id) {
return i;
}
}
throw new RuntimeException("id " + id + " not in ids");
}
Ответ 3
Получите его действие следующим образом: я определил новый styleable
:
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="Label" >
<attr name="android:textColor" />
<attr name="android:textSize" />
<attr name="android:textStyle" />
<attr name="android:typeface" />
</declare-styleable>
</resources>
Затем вот мои styles.xml:
<resources xmlns:android="http://schemas.android.com/apk/res/android">
<style name="Label" parent="@android:style/TextAppearance.Small">
<item name="android:textColor">#12345678</item>
<item name="android:textSize">8dp</item>
<item name="android:textStyle">bold</item>
<item name="android:typeface">serif</item>
</style>
</resources>
И, наконец, тест:
public class TextAppearanceTest extends AndroidTestCase {
public void test() {
TypedArray a = getContext().obtainStyledAttributes(R.style.Label, R.styleable.Label);
assertTrue(a.getColor(R.styleable.Label_android_textColor, -1) != -1);
assertTrue(a.getDimensionPixelSize(R.styleable.Label_android_textSize, -1) != -1);
assertTrue(a.getInt(R.styleable.Label_android_typeface, -1) != -1);
}
}