Как открыть фрагмент «Подробности» по щелчку Recyclerview - PullRequest
0 голосов
/ 19 января 2020

У меня есть этот фрагмент, его viewmodel и его адаптер. Он уже может прослушивать щелчок, но все, что я знаю для этого, это показать тост . Я хочу, чтобы это было go к другому фрагменту, который покажет его "детали", который передает данные по выбранному элементу повторного просмотра в этот фрагмент "деталей". Подсказка: на фрагменте есть //TODO, и мне нужен код для этого.

Вот фрагмент:

class HomeFragment : Fragment(), RecyclerViewClickListener {

    private lateinit var factory: HomeViewModelFactory
    private lateinit var viewModel: HomeViewModel

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        return inflater.inflate(R.layout.fragment_home, container, false)
    }

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

        val api = DormsAPI()
        val repository = DormRepository(api)

        factory = HomeViewModelFactory(repository)
        viewModel = ViewModelProviders.of(this, factory).get(HomeViewModel::class.java)

        viewModel.getDorms()

        viewModel.dorms.observe(viewLifecycleOwner, Observer { dorms ->
            recyclerViewDorms.also{
                it.layoutManager = LinearLayoutManager(requireContext())
                it.setHasFixedSize(true)
                it.adapter = dormAdapter(dorms, this)
            }
        })
    }
    override fun onRecyclerViewItemClick(view: View, dorms: Dorms) {
        when(view.id){
            R.id.button_reserve -> {
                // TODO: Go to new account if not signed up, etc...
                Toast.makeText(requireContext(), "Reserve button clicked", Toast.LENGTH_LONG).show()
            }
            R.id.layoutBox -> {
                // TODO: Go to Dorm Details
                Toast.makeText(requireContext(), "Go to dorm details", Toast.LENGTH_LONG).show()
            }
        }
    }
}

Что касается класса Adapter:

class dormAdapter(
    private val dorms: List<Dorms>,
    private val listener: RecyclerViewClickListener
) : RecyclerView.Adapter<dormAdapter.DormViewHolder>() {

    override fun getItemCount() = dorms.size

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int) =
        DormViewHolder(
            DataBindingUtil.inflate(
                LayoutInflater.from(parent.context),
                R.layout.layout_home,
                parent, false
            )
        )

    override fun onBindViewHolder(holder: DormViewHolder, position: Int) {
        holder.recyclerviewDormBinding.dorm = dorms[position]
        holder.recyclerviewDormBinding.buttonReserve.setOnClickListener {
            listener.onRecyclerViewItemClick(holder.recyclerviewDormBinding.buttonReserve, dorms[position])

        }
        holder.recyclerviewDormBinding.layoutBox.setOnClickListener {
            listener.onRecyclerViewItemClick(holder.recyclerviewDormBinding.layoutBox, dorms[position])
        }
    }

    inner class DormViewHolder(
        val recyclerviewDormBinding: LayoutHomeBinding
    ) : RecyclerView.ViewHolder(recyclerviewDormBinding.root)

}

Наконец, вот ViewModel:

class HomeViewModel(private val repository: DormRepository) : ViewModel() {

    private lateinit var job: Job

    private val _dorms = MutableLiveData<List<Dorms>>()
    val dorms: LiveData<List<Dorms>>
        get() = _dorms

    fun getDorms() {
        job = Coroutines.ioThenMain(
            { repository.getDorms() },
            { _dorms.value = it }
        )
    }

    override fun onCleared() {
        super.onCleared()
        if(::job.isInitialized) job.cancel()
    }
}

РЕДАКТИРОВАТЬ: У меня также есть этот интерфейс, если необходимо:

interface RecyclerViewClickListener {
    fun onRecyclerViewItemClick(view: View, dorms: Dorms)
}

1 Ответ

1 голос
/ 19 января 2020

Поскольку мы добавляем HomeFragment из действия, мы попытаемся создать интерфейс для взаимодействия между действием и фрагментом.

1. Создайте интерфейс

class HomeFragment : Fragment(), RecyclerViewClickListener {
    ...
    ...

    private var callback : Callback? = null

    ...
    ...

    override fun onAttach(context: Context) {
        ...
        // Callback instance is initialized 
        if(context is Callback) callback = context
        else throw RuntimeException("$context must implement Callback")
    }

    ...
    ...

    override fun onDetach() {
        callback = null
    }

    ...
    ...

    override fun onRecyclerViewItemClick(view: View, dorms: Dorms) {
        when(view.id){
            R.id.button_reserve -> {
                // TODO: Go to new account if not signed up, etc...
                Toast.makeText(requireContext(), "Reserve button clicked", Toast.LENGTH_LONG).show()
            }
            R.id.layoutBox -> {
                // Go to Dorm Details
                callback?.onShowDormDetail(dorm)
            }
        }
    }

    ...
    ...

    // This interface will act as mode to communication between
    // activity and fragment
    interface Callback {
        fun onShowDormDetail(dorm: Dorm)
    }
}

2. Реализуйте Callback в вызывающей деятельности

class HomeActivity : AppCompatActivity(), HomeFragment.Callback {

...
...

    override onShowDormDetail(dorm: Dorm) {
        // Add or replace the detail fragment here
    }
}
...