Dagger 2'yi projeme eklemeye çalışıyorum. Parçalarım için ViewModels (AndroidX Architecture bileşeni) enjekte edebildim.
Aynı parçanın (her sekmeler için sadece küçük bir değişiklik) 2 örneğine sahip bir ViewPager var ve her sekmesinde, bir LiveData
veri değişikliği (API) güncellenen almak için gözlemliyorum .
Sorun, api yanıtı geldiğinde ve güncellediğinde LiveData
, şu anda görünür olan parçadaki aynı verilerin tüm sekmelerdeki gözlemcilere gönderilmesidir. (Bence bu muhtemelen kapsamı nedeniyle ViewModel
).
Verilerimi şu şekilde gözlemliyorum:
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
activityViewModel.expenseList.observe(this, Observer {
swipeToRefreshLayout.isRefreshing = false
viewAdapter.setData(it)
})
....
}
ViewModel
S sağlamak için bu sınıfı kullanıyorum :
class ViewModelProviderFactory @Inject constructor(creators: MutableMap<Class<out ViewModel?>?, Provider<ViewModel?>?>?) :
ViewModelProvider.Factory {
private val creators: MutableMap<Class<out ViewModel?>?, Provider<ViewModel?>?>? = creators
override fun <T : ViewModel?> create(modelClass: Class<T>): T {
var creator: Provider<out ViewModel?>? = creators!![modelClass]
if (creator == null) { // if the viewmodel has not been created
// loop through the allowable keys (aka allowed classes with the @ViewModelKey)
for (entry in creators.entries) { // if it's allowed, set the Provider<ViewModel>
if (modelClass.isAssignableFrom(entry.key!!)) {
creator = entry.value
break
}
}
}
// if this is not one of the allowed keys, throw exception
requireNotNull(creator) { "unknown model class $modelClass" }
// return the Provider
return try {
creator.get() as T
} catch (e: Exception) {
throw RuntimeException(e)
}
}
companion object {
private val TAG: String? = "ViewModelProviderFactor"
}
}
Benim ViewModel
böyle bağlıyorum:
@Module
abstract class ActivityViewModelModule {
@MainScope
@Binds
@IntoMap
@ViewModelKey(ActivityViewModel::class)
abstract fun bindActivityViewModel(viewModel: ActivityViewModel): ViewModel
}
@ContributesAndroidInjector
Benim fragman için böyle kullanıyorum :
@Module
abstract class MainFragmentBuildersModule {
@ContributesAndroidInjector
abstract fun contributeActivityFragment(): ActivityFragment
}
Ve bu modülleri MainActivity
alt bileşenime şu şekilde ekliyorum :
@Module
abstract class ActivityBuilderModule {
...
@ContributesAndroidInjector(
modules = [MainViewModelModule::class, ActivityViewModelModule::class,
AuthModule::class, MainFragmentBuildersModule::class]
)
abstract fun contributeMainActivity(): MainActivity
}
İşte benim AppComponent
:
@Singleton
@Component(
modules =
[AndroidSupportInjectionModule::class,
ActivityBuilderModule::class,
ViewModelFactoryModule::class,
AppModule::class]
)
interface AppComponent : AndroidInjector<SpenmoApplication> {
@Component.Builder
interface Builder {
@BindsInstance
fun application(application: Application): Builder
fun build(): AppComponent
}
}
Ben böyle genişletiyor DaggerFragment
ve enjekte ediyorum ViewModelProviderFactory
:
@Inject
lateinit var viewModelFactory: ViewModelProviderFactory
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
....
activityViewModel =
ViewModelProviders.of(this, viewModelFactory).get(key, ActivityViewModel::class.java)
activityViewModel.restartFetch(hasReceipt)
}
key
iki parça için farklı olacaktır.
Sadece geçerli parçanın gözlemcisinin güncellendiğinden nasıl emin olabilirim.
DÜZENLEME 1 ->
Hata ile örnek bir proje ekledim. Sorun yalnızca özel bir kapsam eklendiğinde gerçekleşiyor gibi görünüyor. Lütfen buradan örnek projeye göz atın : Github link
master
şube sorunu ile uygulama vardır. Herhangi bir sekmeyi yenilerseniz (yenilemek için kaydırın) güncellenen değer her iki sekmeye de yansıtılır. Bu sadece ona özel bir kapsam eklediğimde oluyor ( @MainScope
).
working_fine
şube hiçbir özel kapsam ve onun çalışma para cezası ile aynı app vardır.
Soru net değilse lütfen bize bildirin.
working_fine
şubeden yaklaşımı kullanmayacaksın ? Neden kapsama ihtiyacınız var?