Подтвердить что ты не робот

Попытка получить значения атрибута в коде возвращает неверные значения

Я хочу извлечь несколько атрибутов из ресурса стиля (интересуются только атрибутами, входящими в группу 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)

4b9b3361

Ответ 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);
    }
}