- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如何避免将FragmentX绑定(bind)到Fragment
我有几个文件,在这些文件中我只声明了FragmentX
与Fragment
(或ActivityX
与Activity
)的绑定(bind),以便能够将对象作为基类依赖项注入(inject)。
这些文件看起来像这样
@Module
abstract class FragmentXModule {
@Binds
@FragmentScoped
internal abstract fun bindFragment(fragmentX: FragmentX): Fragment
}
最佳答案
更新:实际上要容易得多!
我写了一个很长的答案,就是如果不复制所有代码,实际上是不可能的,那实际上是不可能的。您可以阅读下面的旧答案以供参考,我将在此处仅包括简单的解决方案。
事实证明,存在AndroidInjector.Factory
接口(interface)和AndroidInjector.Builder
类是一个很好的理由。我们可以自己实现接口(interface),而使用我们的构建器!这样,我们仍然可以继续使用Dagger Android部件来注入(inject)我们的组件,而无需自己重新创建。
不同的组件可以使用不同的构建器,最后它们只需要实现AndroidInjector.Factory<T>
即可。以下构建器显示了绑定(bind)类型和一个父类(super class)型的通用方法。
abstract class SuperBindingAndroidInjectorBuilder<S, T : S> : AndroidInjector.Factory<T> {
override fun create(instance: T): AndroidInjector<T> {
seedInstance(instance)
superInstance(instance)
return build()
}
// bind the object the same way `AndroidInjector.Builder` does
@BindsInstance
abstract fun seedInstance(instance: T)
// _additionally_ bind a super class!
@BindsInstance
abstract fun superInstance(instance: S)
abstract fun build(): AndroidInjector<T>
}
AndroidInjector.Builder
,它也允许我们绑定(bind)父类(super class)型。
@Subcomponent
interface MainActivitySubcomponent : AndroidInjector<MainActivity> {
@Subcomponent.Builder
abstract class Builder : SuperBindingAndroidInjectorBuilder<Activity, MainActivity>()
}
Activity
和
MainActivity
。
Is it possible to avoid those file creation repeat and group all bindings in one file?
Component.Builder
。 (是的,您还可以向构建器添加一个带有构造函数参数的模块,但这具有相同的效果并导致更多的代码)
@BindsInstance abstract fun activity(instance: Activity)
添加到
Subcomponent.Builder
中,并在构造组件时将其传递进来。如果您想使用AndroidInjection,则我们需要做更多的事情,我将在下面的文章中详细介绍。
@ContributesAndroidInjector
或
AndroidInjection.inject()
...
@ContributesAndroidInjector
@ContributesAndroidInjector
将为我们生成烦人的可编写样板,但我们需要修改此生成的代码。特别是,我们需要使用组件实现的不同接口(interface),唯一的选择就是自己编写样板。是的,我们当然可以创建自己的AnnotationProcessor来生成所需的样板,但这超出了此答案的范围。
The following part is outdated, but I will leave it as reference to what actually goes on within AndroidInjection. Please use the solution presented above if you want to add additional bindings.
SpecificActivity
本身绑定(bind)到了图形,但是不允许我们将其视为
Activity
。为此,我们将必须使用自己的类并将其绑定(bind)为
Activity
。也许Dagger将来会获得这样的功能。
@ContributesAndroidInjector
为我们生成的默认设置开始。这是您应该熟悉的一个。 (如果您尚未使用AndroidInjection,请不用担心,我们将在下一步中创建自己的设置)
@Component(modules = [AndroidSupportInjectionModule::class, ActivityModule::class])
interface AppComponent {
fun inject(app: App)
@Component.Builder
interface Builder {
@BindsInstance fun app(app: App) : AppComponent.Builder
fun build(): AppComponent
}
}
@Module(subcomponents = [MainActivitySubcomponent::class])
internal abstract class ActivityModule {
@Binds
@IntoMap
@ActivityKey(MainActivity::class)
internal abstract fun bindMainActivityFactory(builder: MainActivitySubcomponent.Builder): AndroidInjector.Factory<out Activity>
}
@Subcomponent
interface MainActivitySubcomponent : AndroidInjector<MainActivity> {
@Subcomponent.Builder
abstract class Builder : AndroidInjector.Builder<MainActivity>()
}
MainActivity
,甚至将其绑定(bind)到模块中的
Activity
上,但这不是我们想要的。我们希望这种绑定(bind)是自动化的。让我们看看是否可以做得更好。
AndroidInjection.inject()
。相反,我们需要创建自己的接口(interface)。本着众多Android库的精神,我将我的接口(interface)称为
AwesomeActivityInjector
。我将保持简短和简单,但是您可以阅读
AndroidInjector以获取更多信息,而我基本上只是复制了这些信息。
activity(activity : Activity)
也将允许我们将 Activity 绑定(bind)到组件。
interface AwesomeActivityInjector<T : Activity> {
fun inject(instance: T)
interface Factory<T : Activity> {
fun create(instance: T): AwesomeActivityInjector<T>
}
abstract class Builder<T : Activity> : AwesomeActivityInjector.Factory<T> {
override fun create(instance: T): AwesomeActivityInjector<T> {
activity(instance) // bind the activity as well
seedInstance(instance)
return build()
}
@BindsInstance
abstract fun seedInstance(instance: T)
@BindsInstance
abstract fun activity(instance: Activity)
abstract fun build(): AwesomeActivityInjector<T>
}
}
AndroidInjector
替换为
AwesomeActivityInjector
。
@Module(subcomponents = [MainActivitySubcomponent::class])
internal abstract class ActivityModule {
@Binds
@IntoMap
@ActivityKey(MainActivity::class)
internal abstract fun bindMainActivityFactory(builder: MainActivitySubcomponent.Builder): AwesomeActivityInjector.Factory<out Activity>
}
@Subcomponent
interface MainActivitySubcomponent : AwesomeActivityInjector<MainActivity> {
@Subcomponent.Builder
abstract class Builder : AwesomeActivityInjector.Builder<MainActivity>()
}
class App : Application() {
@Inject
lateinit var awesomeInjectors: Map<Class<out Activity>, @JvmSuppressWildcards Provider<AwesomeActivityInjector.Factory<out Activity>>>
}
object AwesomeInjector {
fun inject(activity: Activity) {
val application = activity.application as App
val factoryProviders = application.awesomeInjectors
val provider = factoryProviders[activity.javaClass] as Provider<AwesomeActivityInjector.Factory<out Activity>>
@Suppress("UNCHECKED_CAST")
val factory = provider.get() as AwesomeActivityInjector.Factory<Activity>
factory.create(activity).inject(activity)
}
}
AwesomeActivityInjector
Activity
和
MainActivity
。
class MainActivity : AppCompatActivity() {
@Inject
lateinit var mainActivity: MainActivity
@Inject
lateinit var activity: Activity
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
AwesomeInjector.inject(this)
}
}
MainActivity
绑定(bind)为
Activity
。我写了这个答案,以举例说明AndroidInjection的工作方式以及如何适应和修改它。
关于android - 如何避免将FragmentX绑定(bind)到Fragment,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49072558/
我有一个像这样的 fragment 栈 F1 -> F2 -> F3 -> F4 -> F5 现在我需要删除 F2、F3、F4 fragment 。 我需要如果我从 F5 fragment 按下后退按
我需要帮助来理解以下场景的工作原理以及如何实现结果。 我有一个名为 ShoppingCart 的类。它有一个名为 addItemsToShoppingCartFromPreviousOrder 的方法
我有一个包含 ViewPager 的 fragment 。这个 ViewPager 显然是由其他 Fragments 填充的。 我的问题是,ViewPager 中的 Fragment 是否有任何方法(
所以我正在实现一个 FragmentActivity 并尝试添加一个 fragment ,但是我遇到了多个问题。我以前做过这个,实际上我使用的代码与上一个项目(它工作的地方)相同,但由于某种原因它在这
Closed. This question needs details or clarity。它当前不接受答案。
我想将 Android X fragment (androidx.fragment.app.Fragment) 转换为 Android native fragment (android.app.Fra
假设我有一个包含 10 列文本类型 (20) 的 SQLite 表。 ListFragment 将从数据库中提取 4 列并使用 SimpleCursorAdapter 显示在列表中。 选择后,List
我有一个对话框 fragment ,我为延迟初始化创建了一个类。当我显示对话框时,它显示正常。但是,当我关闭对话框时,它崩溃的原因是: fragment 与 fragment 管理器无关。 我也尝试过
我想在 View 分页器更改为 fragment 时刷新该 fragment 。 package com.mcivisoft.rcbeam; import android.os.Bundle; imp
我有一个名为的 Activity MainActivity 我在容器 R.id.mainContainer 中添加了 fragment “BenefitFragment”。 在 BenefitFrag
所以我有这个 ClientListView,效果很好,可以显示客户,我可以单击一个客户并在右侧获取他们的详细信息(在我的第二个 fragment 中)。由此处的布局定义: 这很好用,但后来我
我试过这段代码,但点击第一个 fragment 中的按钮并没有改变第二个 fragment 中的字符串值。 这是第一个 fragment 的 kotlin 文件。它有两个按钮,点击它们可以更改字符串值
我有以下情况:我打开 fragment A 和目标,通过按钮的单击事件转到 fragment B。当我在 fragment B 中并点击后退按钮(为了返回 fragment A) 我想将一些参数传递给
我制作了一个 NavigationDrawer fragment ,其中包含 Home、Settings、Feedback 等项目。 home 项在点击 home 时应该打开,home 是在应用程序启
我正在一个接一个地替换 2 个 fragment ,两个 fragment 都有不同的选项菜单。当我替换第二个 fragment 时,它也显示第一个 fragment 的菜单。 setHasOptio
我有问题: android.app.Fragment$InstantiationException: Unable to instantiate fragment ${packageName}.${a
我正在研究 fragment 转换。当我用第二个 fragment 替换第一个 fragment 时,它出现在第一个 fragment 的下方。我希望它移动到第一个 fragment 之上。我该怎么做
我在抽屉导航里总共有 12 个 fragment 。每个 fragment 都有 volley 方法。每个 fragment 都显示自己的 Volley 响应,除了 position = 1 和 po
我在我的 Activity 中使用了两个 fragment 。最初我将向 Activity 添加一个 fragment 。然后在第一个 fragment 中使用监听器我想用第二个 fragment 替
我正在实现一个“fragments-101”程序,当相应的按钮被点击时我会替换一个 fragment 。然而,下面的事情发生了: 为什么会这样?为什么初始 fragment 没有被完全替换?MainA
我是一名优秀的程序员,十分优秀!