Способы инжектить ViewModel с помощью Dagger: что может пойти не так

image-loader.svg

Инъекция зависимостей во ViewModel — очень популярная тема для статей по всему интернету. Давайте посмотрим, какие проблемы могут скрывать популярные подходы, и разберемся, есть ли способ инжектить ViewModel с помощью Dagger без огромного количества кода или потерь валидации графа зависимостей во время компиляции.

Disclaimer: чтобы разобраться в содержании этой статьи, вам потребуется знание Dagger.

Основная сложность использования DI с ViewModel заключается в том, что при создании ViewModel должна так или иначе проходить через «ViewModelProvider (this, factory).get (YourViewModel: class.java)». Этот метод может быть скрыт внутри делегата «by viewModels { factory }» или вызван напрямую. Без этого ViewModel не будет сохранятся при повороте экрана, а метод onCleared () не будет вызываться, когда ViewModel больше не нужна.

Чтобы сделать примеры как можно проще, я предположу, что у нас есть один компонент AppComponent. Но почти все примеры можно адаптировать к архитектуре с Subcomponent для каждой ViewModel или одним Subcomponent на все вьюмодели.

@Component(modules = [...])
interface AppComponent {
    ...
    fun myViewModel(): MyViewModel
}

В большинстве примеров мы будем использовать такую ViewModel:

class MyViewModel @Inject constructor(
    val repository: Repository
) : ViewModel() {
    ...
}

Repository предоставляется одним из модулей в AppComponent или просто имеет конструктор с аннотацией @Inject.

Также я предполагаю, что мы можем легко получить AppComponent внутри фрагмента, используя метод:

fun Fragment.getAppComponent(): AppComponent = 
(requireContext() as MyApplication).appComponent

Теперь давайте посмотрим на существующие подходы и разберемся, какие проблемы они скрывают.

1. Map, Provider> в ViewModelProvider.Factory (с мультибиндингом или без)

Есть несколько вариантов реализации такого подхода. Самый простой — инжектить провайдеры в фабрику вьюмоделей и там собирать их в Map вручную:

class ViewModelFactory @Inject constructor
    myViewModelProvider: Provider
) : ViewModelProvider.Factory {
    private val providers = mapOf, Provider>(
            MyViewModel::class.java to myViewModelProvider
    )

    override fun  create(modelClass: Class): T {
        return providers[modelClass]!!.get() as T
    }
}

Добавляем фабрику в компонент:

@Component
interface AppComponent {
    fun viewModelsFactory(): ViewModelFactory
}

Теперь мы можем создать вьюмодель внутри фрагмента или активити:

private val viewModel: MyViewModel by viewModels { 
    getAppComponent().viewModelsFactory() 
}

Этот же подход можно реализовать, используя аннотации @IntoMap и @ClassKey (VM: class) для мультибайндинга, но суть будет та же.

Такой подход работает и позволяет заинжектить вьюмодель в пару строк, но у него есть и определенные ограничения:

  1. Фабрика становится сервис-локатором. Это значит, что если мы забываем добавить несколько строк в фабрику (или модуль, если используем мультибайндинг) для новой вьюмодели, то получаем исключение во время выполнения приложения без какой-либо индикации во время компиляции. Обнаружение проблем во время компиляции — это одно из главных преимуществ Dagger, и не хотелось бы его терять.

  2. Мы не можем передавать параметры во вьюмодель из фрагмента или активити. Этот подход не позволяет использовать @AssistedInject, хотя во всех вьюмоделях для однообразных параметров вроде SavedStateHandle можно использовать Subcomponent.

2. Используем Hilt

Hilt — это отличный инструмент от Google. С ним можно обойтись меньшим количеством кода. Пока мы не передаем никаких дополнительных параметров из фрагмента во вьюмодель, этот инструмент работает как часы:

@HiltViewModel
class MyViewModel @Inject constructor
        savedStateHandle: SavedStateHandle
        private val repository: Repository,
) : ViewModel()

Во фрагменте нам нужна будет только одна строчка (кроме необходимой аннотации):

private val viewModel: MyViewModel by viewModels()

Само собой это будет работать только в том случае, если корректно настроить Hilt, но для этого есть множество статей и официальная инструкция. Обратите внимание: если забыть @HiltViewModel, то приложение «упадет» во время выполнения, а не во время компиляции.

Но если мы хотим передать что-то из фрагмента во вьюмодель, то придется инжектить AssistedFactory во фрагмент и создавать фабрику вьюмоделей. ViewModel в этом случае может выглядеть примерно так:

class MyViewModel @AssistedInject constructor(
        @Assisted savedStateHandle: SavedStateHandle,
        private val repository: Repository,
        @Assisted private val screenId: String,
) : ViewModel() {

    @AssistedFactory
    interface Factory {
        fun build(stateHandle: SavedStateHandle, screenId: String): MyViewMode
    }
}

Нам также понадобится универсальная фабрика вьюмоделей, просто чтобы избежать повторяющегося кода:

class LambdaFactory(
        savedStateRegistryOwner: SavedStateRegistryOwner
        private val create: (handle: SavedStateHandle) -> T
): AbstractSavedStateViewModelFactory(savedStateRegistryOwner, null) {
    override fun  create(key: String, modelClass: Class, handle: SavedStateHandle): T {
        return create.invoke(handle) as T
    }
}

Теперь мы можем инжектить фабрику во фрагмент и использовать ее для создания вьюмодели:

@Inject
lateinit var factory: MyViewModel.Factory

private val viewModel: MyViewModel by viewModels {
    LambdaFactory(this) { stateHandle ->
        factory.build(stateHandle, screenId = "something")
    }
}

В этом случае мы теряем некоторые преимущества Hilt и получаем что-то больше похожее на старый добрый Dagger с дополнительными шагами. Если это вас устраивает или вам не нужно ничего передавать во вьюмодель из фрагмента, то Hilt будет для вас отличным решением.

3. Получаем ViewModel из DI и передаем ссылку в фабрику во viewModels-делегате

private val viewModel: MainViewModel by viewModels {
    Factory(getAppComponent().myViewModel())
}

class Factory(private val viewModel: T) : ViewModelProvider.Factory {
    override fun  create(modelClass: Class): T {
        return viewModel as T
    }
}

Такой подход не будет работать, потому что лямбда, переданная во viewModels, будет вызываться при каждом повороте экрана и создавать новые экземпляры вьюмодели. Как ни странно, я видел такой подход в статье где-то на просторах интернета.

Метод viewModelComponent ().myViewModel (), который вызывается при каждом повороте экрана, приведет к тому, что вьюмодели будут множиться. Если мы используем какие-то ресурсы внутри вьюмодели или запускаем корутины в конструкторе, то эти ресурсы и контекст для корутин не будут чиститься для всех вьюмоделей, кроме первой.

Даже если бы этот подход работал, есть шанс, что кто-нибудь вызовет ViewModelProvider ().get () напрямую и получит тот же самый результат.

4. Передаем лямбду для создания ViewModel в фабрику

private val viewModel: MainViewModel by viewModels {
    Factory {
        getAppComponent().myViewModel()
    }
}

class Factory(private val create: () -> T) : ViewModelProvider.Factory {
    override fun  create(modelClass: Class): T 
        return create.invoke() as T
    }
}

В принципе, этот подход работает, но в нем есть скрытая опасность. Допустим, мы используем ViewModel для сохранения какого-то утилитарного класса при повороте экрана (например, Router), а этот класс имеет конструктор, помеченный аннотацией @Inject, и наследует от ViewModel. Тогда мы можем, не глядя в код Router, добавить его как параметр в конструктор вьюмодели:

// Without reading a file with Router we wouldn’t know that it extends ViewModel.
class Router @Inject constructor() : ViewModel()

class MyViewModel @Inject constructor(
    private val router: Router
) : ViewModel()

Что произойдет в таком случае? Router будет создан вместе с вьюмоделью и заинжекчен в ее конструктор, ничего необычного. Но когда будет вызван метод onCleared () вьюмодели, этот же метод не будет вызван для Router. Это может потенциально привести к утечке памяти или еще более неприятным и сложным к поимке багам.

Обратите внимание, что то же самое может произойти, если заинжектить и более очевидную вьюмодель внутрь вьюмодели. Не очень корректно использовать их таким образом, но лучший подход не оставляет места для ошибки, иначе мы бы использовали Koin или другой сервис-локатор вместо Dagger и не беспокоились бы о таких вещах.

Как же избежать этой проблемы? Например, использовать @AssistedInject.

5. Используем @AssistedInject

Если мы договоримся всегда использовать @AssistedInject для классов, наследующих от ViewModel, то указанная выше проблема не возникнет, а также у нас будет возможность передавать во вьюмодель дополнительные параметры.

Давайте немного подкорректируем нашу вьюмодель, чтобы поддержать assisted injection:

class MyViewModel @AssistedInject constructor(
        @Assisted savedStateHandle: SavedStateHandle
) : ViewModel() {

    @AssistedFactory
    interface Factory {
        fun create(savedStateHandle: SavedStateHandle): MyViewModel
    }
}

Подготовим фабрику, аналогичную предыдущему примеру:

class Factory(
        savedStateRegistryOwner: SavedStateRegistryOwner,
        private val create: (stateHandle: SavedStateHandle) -> T
) : AbstractSavedStateViewModelFactory(savedStateRegistryOwner, null) {

    override fun  create(key: String, modelClass: Class, handle: SavedStateHandle): T {
        return create.invoke(handle) as T
    }
}

И один метод, чтобы создавать «ленивый» делегат с фабрикой:

inline fun  Fragment.lazyViewModel(
        noinline create: (stateHandle: SavedStateHandle) -> T
) = viewModels { 
    Factory(this, create)
}

Поменяем AppComponent:

@Component
interface AppComponent {
    fun myViewModel(): MyViewModel.Factory
}

И, наконец, мы можем получить вьюмодель во фрагменте:

private val viewModel: MyViewModel by lazyViewModel { stateHandle ->
    appComponent().myViewModel().create(stateHandle)
}

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

6. Бонус

В моей предыдущей статье я предложил подход, который освобождает от необходимости наследовать классы от ViewModel, а также альтернативный способ очистки ресурсов вьюмоделей. Похожий подход используется в этой библиотеке, так что я не один до этого додумался.

Если вы читали статью, то могли заметить, что lazyViewModel чем-то похож на getOrCreatePersisted из той статьи, хотя последний и не возвращает делегат.

Мы могли бы упаковать все зависимости из статьи в один Subcomponent примерно так:

@Subcomponent
interface ViewModelsComponent {

    fun myViewModel(): MyViewModel

    @Subcomponent.Factory
    interface Factory {
        fun create(
                @BindsInstance coroutineScope: CoroutineScope,
                @BindsInstance savedStateHelper: SavedStateHelper,
                @BindsInstance presistentLifecycle: PersistentLifecycle,
        ): ViewModelComponen
    }
}

Добавим функцию для создания сабкомпонента во фрагменте:

private fun Fragment.viewModelsComponent() = getAppComponent().viewModelsSubcomponent().
        .create(persistentCoroutineScope(), savedStateHelper(), persistentLifecycle())

Добавим зависимостей и уберем наследование от ViewModel из нашей вьюмодели:

class MyViewModel @Inject constructor(
        private val coroutineScope: CoroutineScope,
        private val savedStateHelper: SavedStateHelper,
        private val lifecycle: PersistentLifecycle,
)

Упакуем lazy и getOrCreatePersisted в один метод:

inline fun  ViewModelStoreOwner.lazyPersisted(noinline create: () -> T) = lazy { getOrCreatePersisted(create) }

И теперь можем легко создать нашу вьюмодель во фрагменте:

val viewModel by lazyPersisted {
    viewModelsComponent().myViewModel()
}

Таким образом, у нас не будет необходимости использовать @AssistedInject в том случае, когда он не нужен. Все зависимости вьюмодели, которые требуют очищения ресурсов, могут сами разобраться с ними, приняв в конструктор PersistentLifecycle в качестве параметра. Также наша вьюмодель больше не зависит напрямую от фреймворка, хотя уйти в Kotlin Multiplatform нам пока не позволит Dagger.

Заключение

Хотя это и не всегда очевидно, есть способы инжектить вьюмодель с помощью Dagger, не терять при этом валидацию графа зависимостей при компиляции и не использовать огромное количества кода. Особенно если избавиться от наследования ViewModel, всегда создавать вьюмодель через @AssistedInject или внимательно следить за тем, чтобы во вьюмодели не инджектились другие вьюмодели.

© Habrahabr.ru