Не удается разрешить конструкцию ViewModelProvider во фрагменте? - PullRequest
4 голосов
/ 05 февраля 2020

Я потратил много времени, пытаясь выяснить, почему в приведенном ниже коде (ближе к концу) я получаю сообщение об ошибке на ViewModelProvider (this). Я также попробовал getActivity () вместо «this», та же проблема. Я получаю ошибку «Не удается разрешить конструктор ...»


import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;



public class ItemSetupFragment extends Fragment {

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_setup, container, false);

        return view;
    }

    @Override
    public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

        ItemSetupFragmentModel model = new ViewModelProvider(this).get(ItemSetupFragmentModel.class);
        model.getKids().observe(this, users -> {
            // update UI
        });
    }
}

Image showing compilation error in Android Studio

Ответы [ 4 ]

7 голосов
/ 05 февраля 2020

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

implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'
implementation "androidx.lifecycle:lifecycle-viewmodel:2.2.0" 

или любая обновленная версия.

Тогда вам следует использовать requireActivity() вместо getActivity(). Таким образом, вы убедитесь, что действие прикреплено и не получает NullPointerException.

ItemSetupFragmentModel model = new ViewModelProvider(requireActivity()).get(ItemSetupFragmentModel.class);
2 голосов
/ 05 февраля 2020

Вы не используете последний выпуск библиотеки, в который был включен конструктор ViewModelProvider(@NonNull ViewModelStoreOwner owner). Вы просматриваете последние документы, но не используете последнюю версию библиотеки ViewModel. Вам нужно использовать

    implementation 'androidx.lifecycle:lifecycle-viewmodel:2.2.0' // For Java

или

    implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.2.0' // For kotlin extension
0 голосов
/ 05 февраля 2020

Самая чистая реализация должна использовать Kotlin для своих расширенных функций. Вы можете создать этот kotlin код в отдельном Kclass или оставить этот ответ для будущих пользователей, которые хотят знать, как это сделать в Kotlin. По сути, мы инициализируем ViewModel с помощью lazy следующим образом:

Убедитесь, что у вас есть эта зависимость:

implementation "androidx.fragment:fragment-ktx:1.2.0"

Создайте эту вспомогательную функцию, которая обращается к внутреннему методу фрагмента-ktx, который позволяет вам создать экземпляр ViewModel от lazy:

@MainThread
inline fun <reified VM : ViewModel> Fragment.fragmentViewModel() =
    createViewModelLazy(
        VM::class,
        { this.viewModelStore },
        { ViewModelFactory(Database.getDatabase(requireContext().applicationContext)) }
    )

Теперь создайте ViewModelFactory, используя этот официальный java пример:

https://github.com/android/architecture-components-samples/blob/master/BasicRxJavaSample/app/src/main/java/com/example/android/observability/ui/ViewModelFactory.java

Или, вот вариант Kotlin:

class ViewModelFactory(private val database: Database?) : ViewModelProvider.Factory {

    override fun <T : ViewModel> create(modelClass: Class<T>): T {

        requireNotNull(database) { "Database must not be null" }

        return when {

            modelClass.isAssignableFrom(ItemSetupFragmentModel::class.java) -> {
                ItemSetupFragmentModel() as T
            }

            else -> {
                throw IllegalArgumentException("Unknown ViewModel class")
            }
        }
    }
}

А теперь go внутри вашего фрагмента и просто инициализируйте вашу ViewModel следующим образом

class ItemSetupFragment : Fragment() {

    private val model by viewModel<ItemSetupFragmentModel>()

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

        model.getKids().observe(this, users -> {
            // update UI
        });
    }
}

Надеюсь, это поможет!

0 голосов
/ 05 февраля 2020

Вы должны создать экземпляр вашей модели представления по:

ItemSetupFragmentModel model = ViewModelProviders.of(this).get(ItemSetupFragmentModel.class);
...