This repository has been archived by the owner on Aug 16, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 261
/
PlantListViewModel.kt
185 lines (166 loc) · 5.92 KB
/
PlantListViewModel.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
/*
* Copyright 2018 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.example.android.advancedcoroutines
import androidx.lifecycle.LiveData
import androidx.lifecycle.MutableLiveData
import androidx.lifecycle.ViewModel
import androidx.lifecycle.asLiveData
import androidx.lifecycle.switchMap
import androidx.lifecycle.viewModelScope
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.Job
import kotlinx.coroutines.channels.ConflatedBroadcastChannel
import kotlinx.coroutines.flow.asFlow
import kotlinx.coroutines.flow.catch
import kotlinx.coroutines.flow.flatMapLatest
import kotlinx.coroutines.flow.launchIn
import kotlinx.coroutines.flow.mapLatest
import kotlinx.coroutines.flow.onCompletion
import kotlinx.coroutines.flow.onEach
import kotlinx.coroutines.launch
/**
* The [ViewModel] for fetching a list of [Plant]s.
*
* The @ExperimentalCoroutinesApi and @FlowPreview indicate that experimental APIs are being used.
*/
@ExperimentalCoroutinesApi
@FlowPreview
class PlantListViewModel internal constructor(
private val plantRepository: PlantRepository
) : ViewModel() {
/**
* Request a snackbar to display a string.
*
* This variable is private because we don't want to expose [MutableLiveData].
*
* MutableLiveData allows anyone to set a value, and [PlantListViewModel] is the only
* class that should be setting values.
*/
private val _snackbar = MutableLiveData<String?>()
/**
* Request a snackbar to display a string.
*/
val snackbar: LiveData<String?>
get() = _snackbar
private val _spinner = MutableLiveData<Boolean>(false)
/**
* Show a loading spinner if true
*/
val spinner: LiveData<Boolean>
get() = _spinner
/**
* The current growZone selection.
*/
private val growZone = MutableLiveData<GrowZone>(NoGrowZone)
/**
* A list of plants that updates based on the current filter.
*/
val plants: LiveData<List<Plant>> = growZone.switchMap { growZone ->
if (growZone == NoGrowZone) {
plantRepository.plants
} else {
plantRepository.getPlantsWithGrowZone(growZone)
}
}
/**
* The current growZone selection (flow version)
*/
private val growZoneChannel = ConflatedBroadcastChannel<GrowZone>()
/**
* A list of plants that updates based on the current filter (flow version)
*/
val plantsUsingFlow: LiveData<List<Plant>> = growZoneChannel.asFlow()
.flatMapLatest { growZone ->
if (growZone == NoGrowZone) {
plantRepository.plantsFlow
} else {
plantRepository.getPlantsWithGrowZoneFlow(growZone)
}
}.asLiveData()
init {
clearGrowZoneNumber()
growZoneChannel.asFlow()
.mapLatest { growZone ->
_spinner.value = true
if (growZone == NoGrowZone) {
plantRepository.tryUpdateRecentPlantsCache()
} else {
plantRepository.tryUpdateRecentPlantsForGrowZoneCache(growZone)
}
}
.onCompletion { _spinner.value = false }
.catch { throwable -> _snackbar.value = throwable.message }
.launchIn(viewModelScope)
}
/**
* Filter the list to this grow zone.
*
* In the starter code version, this will also start a network request. After refactoring,
* updating the grow zone will automatically kickoff a network request.
*/
fun setGrowZoneNumber(num: Int) {
growZone.value = GrowZone(num)
growZoneChannel.offer(GrowZone(num))
// initial code version, remove during flow rewrite
launchDataLoad { plantRepository.tryUpdateRecentPlantsForGrowZoneCache(GrowZone(num)) }
}
/**
* Clear the current filter of this plants list.
*
* In the starter code version, this will also start a network request. After refactoring,
* updating the grow zone will automatically kickoff a network request.
*/
fun clearGrowZoneNumber() {
growZone.value = NoGrowZone
growZoneChannel.offer(NoGrowZone)
// initial code version, remove during flow rewrite
launchDataLoad { plantRepository.tryUpdateRecentPlantsCache() }
}
/**
* Return true iff the current list is filtered.
*/
fun isFiltered() = growZone.value != NoGrowZone
/**
* Called immediately after the UI shows the snackbar.
*/
fun onSnackbarShown() {
_snackbar.value = null
}
/**
* Helper function to call a data load function with a loading spinner; errors will trigger a
* snackbar.
*
* By marking [block] as [suspend] this creates a suspend lambda which can call suspend
* functions.
*
* @param block lambda to actually load data. It is called in the viewModelScope. Before calling
* the lambda, the loading spinner will display. After completion or error, the
* loading spinner will stop.
*/
private fun launchDataLoad(block: suspend () -> Unit): Job {
return viewModelScope.launch {
try {
_spinner.value = true
block()
} catch (error: Throwable) {
_snackbar.value = error.message
} finally {
_spinner.value = false
}
}
}
}