如何使用视图绑定制作 BaseFragment

Ali*_*nov 16 android kotlin jetpack androidx android-viewbinding

我的基础片段

abstract class BaseFragment<ViewModel : BaseViewModel, Binding : ViewBinding> : Fragment() {

    protected abstract val viewModel: ViewModel
    private var _binding: Binding? = null
    protected val binding get() = _binding!!

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?
    ): View? {
        _binding = Binding.inflate(inflater, container, false) //This line not working
        return binding.root
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        initialize()
        setupListeners()
        observe()
    }

    abstract fun initialize()

    abstract fun setupListeners()

    abstract fun observe()

    override fun onDestroyView() {
        super.onDestroyView()
        _binding = null
    }
}
Run Code Online (Sandbox Code Playgroud)

但是这条线不起作用 _binding = Binding.inflate(inflater, container, false)

有一个工作代码,但对我来说这是狗屎编码:

abstract class BaseFragment<ViewModel : BaseViewModel, T : ViewBinding>() : Fragment() {

    private var _binding:T? = null
    protected abstract val viewModel: ViewModel
    protected val binding get() = _binding!!


    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?
    ): View? {
        val superclass: Type = javaClass.genericSuperclass!!
        val aClass = (superclass as ParameterizedType).actualTypeArguments[1] as Class<*>
        try {
            val method: Method = aClass.getDeclaredMethod(
                "inflate",
                LayoutInflater::class.java,
                ViewGroup::class.java,
                Boolean::class.javaPrimitiveType
            )
            _binding = method.invoke(null, layoutInflater, container, false) as T

        } catch (e: Exception) {
            e.printStackTrace()
        }
        return binding.root
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        initialize()
        setupListeners()
        observe()
    }

    override fun onDestroyView() {
        super.onDestroyView()
        _binding = null
    }

    abstract fun initialize()

    abstract fun setupListeners()

    abstract fun observe()
}
Run Code Online (Sandbox Code Playgroud)

如何正确地做任何人都有一个干净的解决方案。我知道DataBindingDataBindingUtil但我需要ViewBindingViewBinding是否有类似的东西

我希望你的回答

.

编辑

图书馆: https : //github.com/kirich1409/ViewBindingPropertyDelegate

基础片段:

abstract class BaseFragment<ViewModel : BaseViewModel, Binding : ViewBinding>(
    layoutID: Int
) : Fragment(layoutID) {

    protected abstract val viewModel: ViewModel
    protected abstract val binding: Binding

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        setupViews()
        setupListeners()
        setupObservers()
    }

    abstract fun setupViews()

    abstract fun setupListeners()

    abstract fun setupObservers()
}
Run Code Online (Sandbox Code Playgroud)
class Fragment : BaseFragment<BaseViewModel, FragmentBinding>(
    R.layout.fragment
) {

    override val viewModel: BaseViewModel by viewModels()
    override val binding: FragmentBinding by viewBinding() // this is from library

    override fun setupViews() {
      
    }

    override fun setupListeners() {

    }

    override fun setupObservers() {
        
    }
}
Run Code Online (Sandbox Code Playgroud)

Jon*_*-sz 26

基础片段.kt

typealias Inflate<T> = (LayoutInflater, ViewGroup?, Boolean) -> T

abstract class BaseFragment<VB: ViewBinding>(
        private val inflate: Inflate<VB>
) : Fragment() {
    
    private var _binding: VB? = null
    val binding get() = _binding!!

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        _binding = inflate.invoke(inflater, container, false)
        return binding.root
    }

    override fun onDestroyView() {
        super.onDestroyView()
        _binding = null
    }
Run Code Online (Sandbox Code Playgroud)

HomeFragment.kt

class HomeFragment() : BaseFragment<FragmentHomeBinding>(FragmentHomeBinding::inflate) {

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        binding.homeText.text = "Hello view binding"
    }
}
Run Code Online (Sandbox Code Playgroud)

  • @Jonny-sz 如果您对 BaseFragment 有意见,如何在 HomeFragment 中访问它们? (2认同)

Dea*_*ean 0

abstract class BaseFragment<DB : ViewDataBinding> : Fragment(), ViewContract<DB>, BaseConstant {
    private var isRegistered = false
    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        if (getLayoutResourceId() != 0) {
            val binding: DB = DataBindingUtil.inflate(inflater, getLayoutResourceId(), container, false)
            binding.setLifecycleOwner { lifecycle}
            onBindData(binding)
            return binding.root
        } else {
            throw IllegalArgumentException("layout resource cannot be null")
        }
    }

    override fun onResume() {
        super.onResume()
        if (!isRegistered) {
            kotlinBus.register(this)
            isRegistered = true
        }
    }

    override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
        onInitLiveData()
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        initView()
        initActionView()
    }

    open fun initView() {
        //TODO init view such as adapter, linearLayoutManager
    }

    open fun initActionView() {
        //TODO init action for view such as onClick
    }
    open fun onInitLiveData(){

    }
    override fun onDetach() {
        super.onDetach()
        if (isRegistered) {
            kotlinBus.unregister(this)
            isRegistered = false
        }
    }


    override fun onBindData(binding: DB) {
        //TODO binding data into layout
    }

}
Run Code Online (Sandbox Code Playgroud)

简单的 ViewContract 界面:

interface ViewContract<DB> {
    fun getLayoutResourceId(): Int
    fun onBindData(binding: DB)
}
Run Code Online (Sandbox Code Playgroud)

你的片段类可以是:

class SampleFragment : BaseFragment<FragmentSampleBinding>() {
    private val viewModel by viewModel<HabitsProgressOverallViewModel>()
    @ExperimentalCoroutinesApi
    private val homeViewModel: HomeViewModel by sharedViewModel()
    override fun getLayoutResourceId(): Int {
        return R.layout.fragment_sample
    }

    override fun onBindData(binding: FragmentOverallHabitsCompletionBinding) {
        super.onBindData(binding)
        binding.viewModel = viewModel
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 我需要如何使用 ViewBinding 而不是 DataBinding 来做到这一点 (3认同)