Search code examples
androidandroid-fragmentskotlinandroid-actionmode

Android : startActionMode in fragment only works inside onActivityCreated


I have a fragment that uses a RecyclerView. The fragment implements onClickListener from the adapter class.

Inside the fragment, I want to call startActionMode when I click on the recyclerView item.

startActionMode works inside onActicityCreated method, but not on the onClickListener.

Below is the code :

SongsFrament.kt

package com.akshat.music

import android.content.ContentUris
import android.net.Uri
import android.os.Bundle
import android.provider.MediaStore
import android.view.*
import androidx.fragment.app.Fragment
import androidx.recyclerview.widget.LinearLayoutManager
import kotlinx.android.synthetic.main.fragment_songs.*
import java.io.File

class SongsFragment : Fragment(), SongListAdapter.OnItemClickListener, ActionMode.Callback {

    private val allSongs = ArrayList<Songs>()

    override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
        recyclerView.apply {
            layoutManager = LinearLayoutManager(activity)
            adapter = SongListAdapter(allSongs, SongsFragment())
        }
        activity?.startActionMode(this)   // this works 
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        getAudioDirectories()
    }

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

    override fun onItemClicked(song: Songs, position: Int) {
        activity?.startActionMode(this@SongsFragment) // this doesn't work
    }

    override fun onActionItemClicked(p0: ActionMode?, p1: MenuItem?): Boolean {
        return false
    }

    override fun onCreateActionMode(p0: ActionMode?, p1: Menu?): Boolean {
        activity?.menuInflater?.inflate(R.menu.contextual_action_bar, p1)
        return true
    }

    override fun onPrepareActionMode(p0: ActionMode?, p1: Menu?): Boolean {
        return true
    }

    override fun onDestroyActionMode(p0: ActionMode?) {
    }

    private fun getAudioDirectories() {

        val uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
        val selection = MediaStore.Audio.Media.IS_MUSIC + "!=0"
        val order = MediaStore.Audio.Media.DISPLAY_NAME + " ASC"
        val cursor = activity?.contentResolver?.query(uri, null, selection, null, order)

        cursor?.let {
            it.moveToFirst()
            val pathIndex = it.getColumnIndex(MediaStore.Images.Media.DATA)
            do {
                val path = it.getString(pathIndex)

                val file = File(path)
                if (!file.exists()) {
                    continue
                }
                var songAuth = it.getString(it.getColumnIndex(MediaStore.Audio.Media.ARTIST))
                var songName = it.getString(it.getColumnIndex(MediaStore.Audio.Media.TITLE))
                val albumID = it.getLong(it.getColumnIndex(MediaStore.Audio.Albums.ALBUM_ID))
                val sArtworkUri = Uri.parse("content://media/external/audio/albumart")
                val albumArtUri = ContentUris.withAppendedId(sArtworkUri, albumID)

                var song = Songs(songName, songAuth, albumArtUri)
                allSongs.add(song)

            } while (it.moveToNext())

        }
    }
}

How can I make startActionMode work inside onItemClicked?


Solution

  • You probably forgot to pass your fragment into adpter as OnItemClickListener. Try something like this:

    override fun onActivityCreated(savedInstanceState: Bundle?) {
    super.onActivityCreated(savedInstanceState)
    recyclerView.apply {
        layoutManager = LinearLayoutManager(activity)
        adapter = SongListAdapter(allSongs, SongsFragment()).apply {
            onItemClickListener = this
        }
    }
    activity?.startActionMode(this)   // this works