我有三个带有底部导航栏的片段应用程序,我使用NavigationUI进行切换。
我还有一个viewmodel,用于创建数据(来自 Assets 傻瓜),并且片段观察用于填充回收者 View 的实时数据的数组列表。

我的问题是,每当切换片段时,由于重新创建了片段,我不希望每次重新创建片段时都进行数据检索。因此使用viewmodel。但是在我的情况下,不会保留 View 模型中的数据。
我已经附上了片段和 View 模型代码。我不确定这里出了什么问题。

我尝试记录aaraylist中的条目数,如果我不调用填充arraylist的例程,它将返回0。

歌曲片段

private const val TAG = "Songs Fragment"

class SongsFragment : Fragment(), android.widget.SearchView.OnQueryTextListener {

    private val viewmodel: SongListViewModel by lazy { ViewModelProviders.of(this).get(SongListViewModel::class.java)}

    private val songListAdapter = SongListAdapter(arrayListOf())
    private var raga = "All"

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {

        Log.d(TAG, "onCreateView called")

        return inflater.inflate(R.layout.fragment_songs, container, false)
    }

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

        Log.d(TAG, "onViewCreated called")



        val context: FragmentActivity? = activity

        (activity as AppCompatActivity).supportActionBar?.subtitle = "Songs"

        val assetsPath: AssetManager? = context?.assets
        val assetList = assetsPath?.list("")
        if (assetList != null) {
            for (item in assetList)
                Log.d("SongsFragment", assetsPath.toString() + item)
        }

        arguments?.let {
            raga = SongsFragmentArgs.fromBundle(it).raga
        }

        Log.d("Song Fragment", "Raga passed: " + raga)

        Log.d(TAG, "Number of songs: " + viewmodel.songList.size.toString())

        if (raga == "All") {
            viewmodel.allSongs()
            viewmodel.allSongs(assetsPath!!)
        } else {
            viewmodel.songsForRaga(assetsPath!!, raga)
        }

        Log.d(TAG, ": Songlist size: " + viewmodel.songList.size.toString())


        songList_RecyclerView.apply {
            layoutManager = LinearLayoutManager(context)
            adapter = songListAdapter
        }

        songSearchView.setOnQueryTextListener(this)

        observeViewModel()
    }

    override fun onQueryTextSubmit(query: String): Boolean {

        return false
    }

    override fun onQueryTextChange(newText: String): Boolean {
        Log.i("Song Fragment", "Text change:" + newText.length.toString())

        viewmodel.songSearchFilter(newText)

        return false
    }

    fun observeViewModel() {
        viewmodel.songs.observe(this, Observer { songs ->
            songs?.let {
                Log.d("Song Fragment", "ObserveViewModel")
                songListAdapter.updateSongList(songs)
            }
        })
    }
}

SongListViewModel
private const val TAG = "SongListViewModel"

class SongListViewModel : ViewModel() {

    val songs = MutableLiveData<ArrayList<Song>>()
    var songList = arrayListOf<Song>()

    fun allSongs() {
        songList = getAllSongs()

        songs.value = ArrayList(songList.sortedWith(compareBy({ it.songName })))
    }

    fun allSongs(assetsPath: AssetManager) {

        Log.d(TAG, "allSongs called")
        Log.d(TAG, "Number of songs: " + songList.size.toString())
        getAllSongs(assetsPath)
        songs.value = ArrayList(songList.sortedWith(compareBy({ it.songName })))
    }

    fun songSearchFilter(text: String) {

        var filteredList = arrayListOf<Song>()
        filteredList.clear()

        if (text.length != 0) {
            for (song in songList) {
                if (song.songName.toLowerCase().contains(text)) {
                    filteredList.add(song)
                }

            }
            songs.value = ArrayList(filteredList.sortedWith(compareBy({ it.songName })))

        } else {

            songs.value = ArrayList(songList.sortedWith(compareBy({ it.songName })))
        }


    }

    fun songsForRaga(assetsPath: AssetManager, raga: String) {

        Log.d(TAG, "songsForRaga called")

        var filteredList = arrayListOf<Song>()
        filteredList.clear()
        allSongs(assetsPath)
        for (song in songList) {
            if (song.raga == raga) {
                filteredList.add(song)
            }
        }
        songs.value = ArrayList(filteredList.sortedWith(compareBy({ it.songName })))
    }

fun getAllSongs(assetsPath: AssetManager) {

        Log.d(TAG, "getAllSongs called")
        val bufferedReader = assetsPath.open("test.csv").bufferedReader()
        val lineList = mutableListOf<String>()

        bufferedReader.useLines { lines -> lines.forEach { lineList.add(it) } }
        lineList.forEach {
            val parts = it.split(",")
            songList.add(Song(parts[0], parts[1], parts[2], parts[3], ""))
        }


    }
}

最佳答案

通过传递viewmodelprovider来获取片段的父 Activity ,而不是片段本身,以获取 View 模型。

ViewModelProviders.of( Activity ).get(SongListViewModel::class.java)

10-04 10:32
查看更多