How to pass argument to detail ViewModel via navArgs

4k Views Asked by At

I have a screen with a RecyclerView of podcasts, where when you click one, it takes you to a detail screen for that particular podcast. Using Kodein for ViewModel injection, how can I pass the id of the podcast that was clicked from the list fragment to the detail fragment's ViewModel so that it can be fetched from an API?

The detail ViewModel is structured like this:

class PodcastDetailViewModel internal constructor(
    private val podcastRepository: PodcastRepository,
    podcastId: String = ""
): ViewModel() {
    // viewmodel stuff
}

The detail fragment looks like this:

class PodcastDetailFragment : ScopedFragment(), KodeinAware {
    override val kodein by closestKodein()
    private val args: PodcastDetailFragmentArgs by navArgs()
    private val viewModelFactory: PodcastDetailViewModelFactory by kodein.newInstance { PodcastDetailViewModelFactory(args.podcastId, instance()) }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        viewModel = viewModelFactory.create(PodcastDetailViewModel::class.java)
    }

    // other stuff
}

And this is how I'm navigating to the detail screen from the list:

private fun navigateToPodcastDetailFragment(podcastId: String) {
        val args = Bundle()
        args.putString("podcast_id", podcastId)

        val directions =
            TopPodcastsFragmentDirections.viewPodcastDetails(
                podcastId
            )
        val extras = FragmentNavigatorExtras(
            podcast_image to "podcastImage_$podcastId"
        )

        Navigation.findNavController(requireActivity(), R.id.nav_host_fragment)
            .navigate(directions, extras)
    } 

This is how I'm binding it:

bind() from provider { PodcastDetailViewModelFactory(instance(), instance()) }

I'm not sure how to bind that string parameter in the ViewModelFactory constructor, or how else to pass the data there, so any help would be much appreciated.

4

There are 4 best solutions below

1
On

You could use a factory for your bindings:

bind() from factory { podcastId: String -> 
    PodcastDetailViewModelFactory(podcastId, instance()) 
}

calling it on-site with:

private val viewModelFactory: PodcastDetailViewModelFactory by instance(arg = args.podcastId) 

hope this helps.

0
On

I fixed this by switching to a factory binding, like romainbsl suggested, but the call in the fragment was a little different.

The binding became:

bind() from factory { podcastId: String -> 
    PodcastDetailViewModelFactory(podcastId, instance()) 
}

and the on-site call became:

private val viewModelFactory: PodcastDetailViewModelFactory by instance(arg = args.podcastId)
0
On

Try this :

private val viewModelFactory: PodcastDetailViewModelFactory by instance {arg = args.podcastId}

to instantiate by Lazy because navArgs() is an implementation of Lazy used by android.app.Activity.navArgs and androidx.fragment.app.Fragment.navArgs.

0
On

Try this

private val viewModel by viewModels<PodcastDetailViewModel> {
  val viewModelFactory: PodcastDetailViewModelFactory by instance(arg = args.podcastId)
  viewModelFactory
}

OR

private val viewModel by activityViewModels<PodcastDetailViewModel> {
  val viewModelFactory: PodcastDetailViewModelFactory by instance(arg = args.podcastId)
  viewModelFactory
}