如何将抽象与 ViewBinding 与基本活动一起使用?

Mir*_*aig 11 android abstraction generic-programming android-viewbinding

我正在制作一个基类,以便所有孩子的绑定都将设置在基类中

我已经做到了这一点

abstract class BaseActivity2<B : ViewBinding?, T : BaseViewModel?> : AppCompatActivity() {
    private var viewBinding: B? = null
    private var baseViewModel: T? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}
Run Code Online (Sandbox Code Playgroud)

但我无法在 oncreat() 中绑定视图,通常我们将视图绑定中的布局绑定为

binding = ActivityLoginBinding.inflate(layoutInflater)
        setContentView(binding.root)
Run Code Online (Sandbox Code Playgroud)

但我正在寻找基本活动的通用方式

Sin*_*tem 9

您可以在构造函数中声明一个 lambda 属性来创建绑定对象

abstract class BaseActivity<B : ViewBinding>(val bindingFactory: (LayoutInflater) -> B) : AppCompatActivity() {
    private lateinit var binding: B

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = bindingFactory(layoutInflater)
        setContentView(binding.root)
    }
}
Run Code Online (Sandbox Code Playgroud)

您还可以定义binding为惰性属性

private val binding: B by lazy { bindingFactory(layoutInflater) }
Run Code Online (Sandbox Code Playgroud)

然后你需要在你的活动中覆盖任何东西

class MainActivity : BaseActivity<ActivityMainBinding>(ActivityMainBinding::inflate)
Run Code Online (Sandbox Code Playgroud)

  • 您还可以通过惰性委托定义“绑定”。 (3认同)

Mir*_*aig 8

其他答案也可以解决问题,但我想以一种干净的方式来做。

我的基类

abstract class BaseVMActivity<VM : ViewModel, B : ViewBinding> : BaseActivity() {

    lateinit var viewModel: VM
    lateinit var binding: B


    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        viewModel = ViewModelProvider(this, factory).get(getViewModelClass())
        binding = getViewBinding()
        setContentView(binding.root)
    }

    private fun getViewModelClass(): Class<VM> {
        val type = (javaClass.genericSuperclass as ParameterizedType).actualTypeArguments[0]
        return type as Class<VM>
    }

    abstract fun getViewBinding(): B

}
Run Code Online (Sandbox Code Playgroud)

我的活动:

class MainActivity : BaseVMActivity<AppViewModel, ActivityMainBinding>() {
    override fun getViewBinding() = ActivityMainBinding.inflate(layoutInflater)
   
}
Run Code Online (Sandbox Code Playgroud)

现在我可以直接访问 viewModel 或绑定:

fun dummy(){
        binding.bvReport.text = viewModel.getReportText()
    }
Run Code Online (Sandbox Code Playgroud)

  • 你能用JAVA来做这个吗,提前谢谢 (3认同)

Rez*_*eza 6

我认为在子活动中覆盖绑定对象 getter 更干净。所以:

abstract class VBActivity<VB : ViewBinding> : AppCompatActivity() {
  protected abstract val binding: VB

  override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(binding.root)
  }
}
Run Code Online (Sandbox Code Playgroud)

假设 MainActivity 类似于:

class MainActivity : VBActivity<ActivityMainBinding>() {

  override val binding get() = ActivityMainBinding.inflate(layoutInflater)

}
Run Code Online (Sandbox Code Playgroud)