9

我正在尝试在 Android 应用程序上实现 Hilt,虽然与 Dagger 相比,它很容易实现并删除了很多样板代码,但我想念一些东西,比如构建我自己的组件并自己确定它们的范围,所以我会有我自己的雇佣关系。

直截了当:示例:假设我有一个简单的应用程序,其中包含一个 RecyclerView、Adapter、Acitivity 和一个嵌套在我的 Adapter 中的回调,我将其传递给我的 Adapter 构造函数以检测点击或其他什么,我让我的活动实现那个回调,当然我想注入适配器。

class @Inject constructor (callBack: Callback): RecyclerView.Adapter...

当我让 Hilt 知道我想注入我的适配器时,我需要让 Hilt 知道如何提供所有适配器依赖项——回调。

在 Dagger 中,我可以通过将 Activity 绑定到我的一个模块中的回调来实现这一点:

@Binds fun bindCallback(activity: MyActivity): Adapter.Callback

Dagger 知道如何绑定 Activity(或任何 Activity/Fragment),然后将其链接到该回调,但使用 Hilt 它不起作用。

我怎样才能做到这一点?如何使用 Hilt 提供或绑定 Activity 或 Fragment?

4

2 回答 2

11

解决方案非常简单。

所以几天前我回来看我的问题只是发现仍然没有新的解决方案,所以我尝试了 Bartek 解决方案并无法使其工作,即使它确实工作,干净的 Hilt 代码变得太乱了,所以我做了一点调查并玩了一点,发现解决方案实际上非常简单。

它是这样的:

应用程序:

@HiltAndroidApp
class MyApp: Application()

活动:(实现回调)

@AndroidEntryPoint
class MainActivity : AppCompatActivity(), SomeClass.Callback {

    @Inject
    lateinit var someClass: SomeClass

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)  
    }

    override fun onWhatEver() {
        // implement
    }
}

SomeClass:(带有内部回调)

class SomeClass @Inject constructor(
    private val callback: Callback
) {

    fun activateCallback(){
        callback.onWhatEver()
    }

    interface Callback{
        fun onWhatEver()
    }
}

SomeModule:(将活动提供/绑定到回调)

@Module
@InstallIn(ActivityComponent::class)
object SomeModule{

    @Provides
    fun provideCallback(activity: Activity) =
        activity as SomeClass.Callback
    
}

这就是我们所需要的。我们无法使用 @Bind 将活动绑定到回调,因为它需要显式提供并强制转换为回调,以便应用程序可以构建。

该模块安装在 ActivityComponent 中并且知道一个通​​用的“活动”,如果我们将它转​​换为回调,Hilt 是内容并且活动绑定到回调,并且只要它在 Hilt 就会知道如何提供回调具体活动范围。

多个活动/片段

应用程序:

@HiltAndroidApp
class MyApp: Application()

图书活动:

@AndroidEntryPoint
class BooksActivity : AppCompatActivity(), BooksAdapter.Callback{

        @Inject
        lateinit var adapter: BooksAdapter

        ...

        override fun onItemClicked(book: Book) {...}
    }
}

作者活动:

@AndroidEntryPoint
class AuthorsActivity : AppCompatActivity(), AuthorsAdapter.Callback{

    @Inject
    lateinit var adapter: AuthorsAdapter

    ...

    override fun onItemClicked(author: Author) {...}
}

图书适配器

class BooksAdapter @Inject constructor (
    val callback: Callback
) ... {

    ...

    interface Callback{
        fun onItemClicked(book: Book)
    }
}

作者适配器

class AuthorsAdapter @Inject constructor (
    val callback: Callback
) ... {

    ...

    interface Callback{
        fun onItemClicked(auhtor: Auhtor)
    }
}

作者模块

@Module
@InstallIn(ActivityComponent::class)
object AuthorsModule {
    @Provides
    fun provideCallback(activity: Activity) =
        activity as AuthorsAdapter.Callback
}

书籍模块

@Module
@InstallIn(ActivityComponent::class)
object BooksModule {
    @Provides
    fun provideCallback(activity: Activity) =
        activity as BooksAdapter.Callback
}

模块可以毫无问题地连接到一个模块,只需更改函数的名称。

这当然适用于更多活动和/或多个片段......适用于所有逻辑案例。

于 2020-12-20T21:56:23.567 回答
4

Hilt 可以提供泛型的一个实例Activity(以及Fragment就此而言)作为ActivityComponent(和FragmentComponent分别)内的依赖项。它只是无法提供您特定的实例MyActivity

您仍然可以在 Hilt 中创建自己Component的。您只需要自己管理component实例。添加作为组件构建器的种子数据,您MyActivity应该可以毫无问题地使用。@BindCallback

于 2020-08-14T09:53:17.463 回答