PreferenceFragmentCompat с пользовательским макетом не находит панель инструментов в API 28+ - PullRequest
1 голос
/ 23 февраля 2020

У меня есть одно приложение для занятий, которое использует фрагмент настроек (PreferenceFragmentCompat) с пользовательским макетом (устанавливается через переопределение стиля PreferenceThemeOverlay). Из метода onViewCreated фрагмента я использую setSupportActionBar для ссылки на панель инструментов, которая присутствует в этом макете. Это работает нормально до тех пор, пока он не будет работать на API 28+, после чего ссылка на toolbarSettings будет нулевой и возникнет ошибка.

java.lang.NullPointerException: Attempt to invoke virtual method 'void androidx.appcompat.app.ActionBar.setTitle(java.lang.CharSequence)' on a null object reference
        at androidx.navigation.ui.ActionBarOnDestinationChangedListener.setTitle(ActionBarOnDestinationChangedListener.java:48)
        at androidx.navigation.ui.AbstractAppBarOnDestinationChangedListener.onDestinationChanged(AbstractAppBarOnDestinationChangedListener.java:104)
        at androidx.navigation.NavController.addOnDestinationChangedListener(NavController.java:204)
        at androidx.navigation.ui.NavigationUI.setupActionBarWithNavController(NavigationUI.java:228)
        at androidx.navigation.ui.ActivityKt.setupActionBarWithNavController(Activity.kt:74)
        at androidx.navigation.ui.ActivityKt.setupActionBarWithNavController$default(Activity.kt:89)
        at com.simplenotes.notes.presentation.ui.SettingsFragment.setupActionBar(SettingsFragment.kt:39)
        at com.simplenotes.notes.presentation.ui.SettingsFragment.onViewCreated(SettingsFragment.kt:20)
        at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:332)

Я видел упоминание об изменениях фрагментов пользовательских настроек в этом Однако версия не может увидеть ничего, определяющего c такое поведение. Кто-нибудь знает решение или может указать, что было сделано неправильно?

SettingsFragment.kt

class SettingsFragment : PreferenceFragmentCompat() {

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        setupActionBar()
    }

    private fun setupActionBar() {
        setHasOptionsMenu(true)

        val hostActivity = requireActivity() as AppCompatActivity
        hostActivity.setSupportActionBar(toolbarSettings)
        hostActivity.setupActionBarWithNavController(findNavController())

        val actionBar = hostActivity.supportActionBar
        actionBar?.title = resources.getString(R.string.title_settings)
        actionBar?.setDisplayHomeAsUpEnabled(true)
    }
}

frag_settings. xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <com.google.android.material.appbar.AppBarLayout
        android:id="@+id/appbarSettings"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent">

        <androidx.appcompat.widget.Toolbar
            android:id="@+id/toolbarSettings"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize" />

    </com.google.android.material.appbar.AppBarLayout>

    <FrameLayout
        android:id="@android:id/list_container"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</LinearLayout>

стилей. xml

<resources>
    <style name="Theme.MaterialComponents.DayNight.NoActionBar.Bridge" parent="Theme.MaterialComponents.Light.NoActionBar.Bridge" />

    <style name="AppTheme" parent="Theme.MaterialComponents.DayNight.NoActionBar.Bridge">
        <item name="preferenceTheme">@style/AppTheme.PreferenceThemeOverlay</item>
    </style>

    <style name="AppTheme.PreferenceThemeOverlay" parent="PreferenceThemeOverlay">
        <item name="android:layout">@layout/fragment_settings</item>
    </style>
</resources>

AndroidManifest. xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.simplenotes.notes">

    <application
        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">

        <activity android:name=".presentation.ui.MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>

Зависимости

implementation 'androidx.preference:preference:1.1.0'
implementation 'com.google.android.material:material:1.1.0'

1 Ответ

1 голос
/ 05 марта 2020

В случае, если кто-то столкнулся с этой проблемой, я обнаружил проблему.

В дополнение к файлу styles.xml в вопросе был также файл styles-v27.xml, который использовался для переопределения windowLightNavigationBar attribute ...

<resources>
    <style name="AppTheme" parent="Theme.MaterialComponents.DayNight.NoActionBar.Bridge">
        <item name="android:windowLightNavigationBar">false</item>
    </style>
</resources>

В API v28, хотя это означало, что значение preferenceTheme не было получено. Когда фрагмент настроек был накачан, у него фактически не было пользовательского макета, следовательно, панель инструментов не создавалась.

В styles-v27.xml Я просто повторил атрибут preferenceTheme, как показано ниже, и эй до того, что он отлично работает на v28 и выше.

<resources>
    <style name="AppTheme" parent="Theme.MaterialComponents.DayNight.NoActionBar.Bridge">
        <item name="android:windowLightNavigationBar">false</item>

        <item name="preferenceTheme">@style/AppTheme.PreferenceThemeOverlay</item>
    </style>
</resources>
...