android.view.InflateException при использовании selectableItemBackground

Раздувая мой макет, я получаю это исключение:

E AndroidRuntime: android.view.InflateException: Binary XML file line #11: Binary XML file line #11: Error inflating class <unknown>
E AndroidRuntime:        at android.view.LayoutInflater.inflate(LayoutInflater.java:539)
E AndroidRuntime:        at android.view.LayoutInflater.inflate(LayoutInflater.java:423)
E AndroidRuntime:        at com.myapp.view.MyRecyclerAdapter.onCreateViewHolder(MyRecyclerAdapter:80)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Adapter.createViewHolder(RecyclerView.java:5288)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4551)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4461)
E AndroidRuntime:        at android.support.v7.widget.LinearLayoutManager$LayoutState.next(LinearLayoutManager.java:1962)

В журнале нет "Причины", но я добавил код, чтобы перехватить исключение и вызвать getCause() пока он не возвращает ноль, и вот последовательность событий:

android.view.InflateException: Binary XML file line #11: Binary XML file line #11: Error inflating class <unknown>
android.view.InflateException: Binary XML file line #11: Error inflating class <unknown>
Error: Binary XML file line #11: Error inflating class <unknown>
android.content.res.Resources$NotFoundException: File res/drawable-v11/selectable_list_background.xml from drawable resource ID #0x7f020096
java.lang.UnsupportedOperationException: Failed to resolve attribute at index 0: TypedValue{t=0x2/d=0x7f0100f9 a=-1}

0x7f020096 is selectable_list_background (см. ниже), и TypedValue, на который он ссылается, ?selectableItemBackground,

Я сузил исключение до моего использования ?selectableItemBackground, Более конкретно, я использую CardView:

<android.support.v7.widget.CardView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:foreground="@drawable/selectable_list_background"
    card_view:cardCornerRadius="4dp"
    card_view:cardElevation="4dp"
    >

И это соответствующая часть drawable/selectable_list_background:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:drawable="?selectableItemBackground" />
</selector>

Это занятие использует мою собственную тему, его родитель Theme.AppCompat.Light.NoActionBar,

Этот код раньше работал, но я только через несколько месяцев выкопал этот код, и теперь он падает с исключением. Я предполагаю, что это связано с обновлением библиотеки поддержки до 23.0.1 и целевого SDK до 23, но я еще не переключился на 22, чтобы убедиться в этом.

Все работает нормально, если я уберу ссылку на одну строку ?selectableItemBackground, Я тоже пробовал ?attr/selectableItemBackground а также ?android:attr/selectableItemBackground, но получил те же результаты. (Последнее заставляет меня поверить, что это не может быть проблемой с библиотекой поддержки).

РЕДАКТИРОВАТЬ:

Я посмотрел на это в отладчике, и у меня есть подозрение, что именно этот код в android.content.res.Resourcesвнутри loadDrawable():

Drawable loadDrawable(TypedValue value, int id, Theme theme) throws NotFoundException {
    [...]
    dr = loadDrawableForCookie(value, id, null);

Обратите внимание, что эта функция берет тему, но не передает ее при вызове loadDrawableForCookie(), который является методом, который в конечном итоге вызывает первое исключение:

java.lang.UnsupportedOperationException: Failed to resolve attribute at index 0: TypedValue{t=0x2/d=0x7f0100f9 a=-1}
    at android.content.res.TypedArray.getDrawable(TypedArray.java:867)
    at android.graphics.drawable.StateListDrawable.inflateChildElements(StateListDrawable.java:170)
    at android.graphics.drawable.StateListDrawable.inflate(StateListDrawable.java:115)
    at android.graphics.drawable.Drawable.createFromXmlInner(Drawable.java:1215)
    at android.graphics.drawable.Drawable.createFromXml(Drawable.java:1124)
    at android.content.res.Resources.loadDrawableForCookie(Resources.java:2630)
    at android.content.res.Resources.loadDrawable(Resources.java:2540)
    at android.content.res.TypedArray.getDrawable(TypedArray.java:870)
    at android.view.View.<init>(View.java:4280)

Этот код кажется новым для Android 6.0 - код 5.0 совершенно другой, но тема передается при загрузке отрисовки. И тема необходима AFAICT - рассматриваемый атрибут является частью AppCompatтак что для его решения нужна тема действия.

Это похоже на вопиющую ошибку, поэтому я не уверен, что я на правильном пути. Есть идеи?

3 ответа

Решение

Официальный ответ - вы не можете использовать атрибуты темы для android:drawable или же android:src атрибутов. Просто не возможно, по замыслу. Есть еще один ответ, который исследует ту же проблему и предлагает решение: как ссылаться на атрибуты стиля из прорисовки?

В моем случае, я не мог этого сделать, так как AppCompat's ?selectableItemBackground ссылки на частную коллекцию. Поэтому я выбрал решение кода - я устанавливаю передний план на ?selectableItemBackground пока карта не выбрана, и мой собственный список состояний можно будет нарисовать, когда я нахожусь в режиме выбора.

Это очень уродливое решение, настолько уродливое, что я не собираюсь делиться здесь кодом, но лучшее, что я мог придумать до сих пор. Если у кого-то есть лучшее решение, я хотел бы услышать это.

Как говорит @Artjom, проблема в передаче application context при создании собственного представления. Я также столкнулся с той же проблемой, и я проходил getApplicationContext() вместо activity как context, После прохождения Activity как context проблема решена.

Надеюсь, что это может помочь кому-то еще.

У меня была такая же ошибка при использовании компонента тестирования фрагментов androidx. По умолчанию launchFragmentInContainer() использует тему FragmentScenarioEmptyFragmentActivityTheme, что приводит к ошибке.

Простое решение - использовать собственную тему, расширяющую AppCompat:

launchFragmentInContainer(theme = R.style.AppTheme)

К сожалению, другие ответы не в моем случае. Способ работы для меня - проверить, использует ли ваша деятельность стиль темы AppCompat:

    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
    </style>

Затем используйте тему AppCompat в своей деятельности (во многих отношениях, но я покажу, как использовать Manifest):

<application
        android:name=".Abbott"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">

    <!-- Any configuration inside -->

</application>
Другие вопросы по тегам