Как сделать BaseFragment с привязкой к представлению

Мой BaseFragment:

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
    }
}

Но эта строчка не работает _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()
}

Как это сделать правильно, у каждого есть чистое решение. Я знаю, что DataBinding имеет DataBindingUtil, но мне нужен ViewBinding. Есть ли что-то подобное для ViewBinding

Надеюсь на твои ответы

.

ИЗМЕНИТЬ

Библиотека: https://github.com/kirich1409/ViewBindingPropertyDelegate.

BaseFragment:

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()
}
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() {
        
    }
}

person Alis Abenov    schedule 13.11.2020    source источник
comment
да, пробовал, работает, но можно ли в базовом фрагменте делать все сразу   -  person Alis Abenov    schedule 21.11.2020
comment
Привет, я написал сообщение в блоге, полностью объясняющее привязку представлений, и написал baseclass for activity and fragment checkout ???? Androidbites | ViewBinding   -  person Chetan Gupta    schedule 06.12.2020
comment
Отвечает ли это на ваш вопрос? Как использовать ViewBinding с абстрактным базовым классом   -  person Chetan Gupta    schedule 06.12.2020
comment
Спасибо, но хотелось бы более чистого решения   -  person Alis Abenov    schedule 06.12.2020
comment
Помимо хорошего комментария Четана, есть много результатов Google / образцов Github для вашего конкретного требования.   -  person akubi    schedule 11.01.2021


Ответы (2)


BaseFragment.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
    }

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"
    }
}
person Jonny-sz    schedule 18.02.2021
comment
интересно хммм - person Alis Abenov; 18.02.2021
comment
@ Jonny-sz, если у вас есть представления в BaseFragment, как получить к ним доступ в HomeFragment? - person Edalat Feizi; 14.05.2021
comment
@Edalat Feizi Извините, но я не понимаю вопроса. Пожалуйста, поделитесь примером. - person Jonny-sz; 15.05.2021

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
    }

}

простой интерфейс ViewContract:

interface ViewContract<DB> {
    fun getLayoutResourceId(): Int
    fun onBindData(binding: DB)
}

и ваш класс фрагмента может быть:

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
    }
}
person Dean    schedule 13.11.2020
comment
Мне нужно, как это сделать с помощью ViewBinding, а не DataBinding - person Alis Abenov; 13.11.2020