【问题标题】:Can I use one Factory to bind viewmodel / repository calls with kodein我可以使用一个工厂将视图模型/存储库调用与 kodein 绑定吗
【发布时间】:2019-12-18 11:43:14
【问题描述】:

在这个工厂中,我需要使用 Retrofit 从 api 获取我的数据并将缓存与空间一起存储,我的存储库管理这个应用程序! 我有存储库暂停功能,负责获取我的数据,一些保存/更新数据获取和保存/更新需要不同的值才能运行,我不知道(尚)如何在 Kodein 中配置它 我缺乏解决这个问题的经验,而且我在 Stackoverflow 中找不到任何可以帮助我的东西。

我尝试将变量 ID:String 和编辑的实体 (CampaignEntry) 添加到定义中,它符合但在运行时崩溃

No binding found for bind<CampaignEditViewModelFactory>() with ? { String -> ? }

我的主要应用程序 bind() 使应用程序崩溃

class MarketingApplication : Application(), KodeinAware {
    override val kodein = Kodein.lazy {
        import(androidXModule(this@MarketingApplication))
...
bind() from factory { id: String, campaignEntry: CampaignEntry -> 
            CampaignEditViewModelFactory(id, campaignEntry, instance()) }
...

我的 ViewModel - 必须在一个 ViewModel 中传递由不同调用使用的变量 id 和 campaignEntry 可能是问题 - 但我无法找出正确的解决方案。

class CampaignEditViewModel(
    private val id: String,
    private val campaignEntry: CampaignEntry,
    private val marketingRepository: MarketingRepository
) : ViewModel() {
    val campaignToSave by lazyDeferred { marketingRepository.updateCampaign(campaignEntry) }
    val campaignToEdit by lazyDeferred { marketingRepository.getCampaignById(id) }
}

为了清楚起见,我的lazyDeferred

fun <T> lazyDeferred(block: suspend CoroutineScope.() -> T): Lazy<Deferred<T>> {
    return lazy {
        GlobalScope.async(start = CoroutineStart.LAZY) {
            block.invoke(this)
        }
    }
}

存储库快照

interface MarketingRepository {
...
    suspend fun getCampaignById(campaignId: String): LiveData<CampaignEntry>
    suspend fun updateCampaign(campaignEntry: CampaignEntry): LiveData<CampaignEntry>
...

我像这样从我的片段中调用 Viewmodel

class CampaignEditFragment : ScopedFragment(), KodeinAware {
    override val kodein by closestKodein()
    private val viewModelFactoryInstanceFactory: ((String) -> CampaignEditViewModelFactory) by factory()
...
private fun bindUI() = launch {
        val campaignVM = campaignEditViewModel.campaignToEdit.await()
...
    btn_edit_save.setOnClickListener {it: View
                saveCampaign(it)
...
private fun saveCampaign(it: View) = launch {
        campaignEditViewModel.campaignToSave.await()
    }

最后是 ScopedFragment

abstract class ScopedFragment : Fragment(), CoroutineScope {
    private lateinit var job: Job

    override val coroutineContext: CoroutineContext
        get() = job + Dispatchers.Main

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        job = Job()
    }

    override fun onDestroy() {
        super.onDestroy()
        job.cancel()
    }
}

如果您需要更多代码 - 请询问

【问题讨论】:

    标签: android kotlin mvvm kotlin-coroutines kodein


    【解决方案1】:

    由于您使用 2 个参数进行绑定,因此您需要使用 factory2:

    private val viewModelFactoryInstanceFactory: ((String, campaignEntry) -> CampaignEditViewModelFactory) by factory2()
    

    【讨论】:

    • Thanx 这解决了问题,但也向我展示了另一条路径,我认为在我的应用程序中会更好,我使用 safeArcs 传递整个实体记录而不仅仅是 ID 的路径(获取来自存储库的记录)这样我可以节省可能从 Internet 重新获取数据的延迟,并且只有在编辑后才进行更新(如果数据更改),只需要将参数减少到一个 :)跨度>
    猜你喜欢
    • 1970-01-01
    • 2021-01-19
    • 2015-10-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-03-02
    • 2011-12-29
    相关资源
    最近更新 更多