Let's talk.
First is Jetpack Compose UI. It's cool, it's new. I've migrated all supported pyamsoft Android applications to use it. It comes with some new ways of doing things. This is largely a nerd corner post.
First, the move to Jetpack Compose UI sees the deprecation of a number of PYDroid things. The entire UiView system is deprecated, along with UiViewEvents and UiControllerEvents. The UiViewModel is deprecated. The only thing that survives the migration without deprecation is the UiViewState. Let me tell you why.
First, the UiViewEvent and UiControllerEvent and UiView existed as a hacky way to create a React-like render loop, where a ViewModel would control and constantly push an entire view state. In Compose UI, this is all done for us, so all of this ceremony around binding to the normal Android view system is gone. Write Composables as you normally would, pass then props and callbacks as you normally would, and have those callbacks call ViewModel methods as you normally would. This effectively migrates PYDroid-Arch from an MVI to a better MVVM architecture.
Second is ViewModels. I am going to put forward the controversial claim: You no longer need the Android Jetpack ViewModel. PYDroid uses completely independent simple ViewModels called ViewModelers
instead of Jetpack ViewModel and it's weird ViewModelStore and other initialization quirks. Why do this? Well, lets examine what exactly we need from a ViewModel that we do not otherwise already have.
ViewModel comes with 2 very important things for us - it survives configuration changes, and with SavedStateHandle can also persist data across process death. As a trade off, we receive the ViewModel class which must be injected by a ViewModelStore and a Factory, and is difficult to configure with things like scope and injection. What if we could remove all this extra baggage, and be left with a ViewModel that still survives configuration changes and process persistence? What if we could manage scoping and injection simply through Dagger as we do for every other class in our project? Well lets see if Compose UI can't help us change things around.
Let's examine configuration changes first. What if I told you that Compose UI can handle all configuration changes on it's own due to the new system? Yes, thats right. You can finally add the configChanges
line to your manifest as follows:
android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
Also, be sure to override in your Fragments or Activities
override fun onConfigurationChanged(newConfig: Configuration) {
super.onConfigurationChanged(newConfig)
mySyncDarkThemeFunction()
myDialogSizingFunction()
myComposeView.apply {
if (isAttachedToWindow) {
disposeComposition()
createComposition()
}
}
}
This way your components will guarantee that Compose UI is kept in sync with the rest of the system like dark theme and dialog sizes and window sizes and whatnot.
This is massively important because now 1 of the two benefits of ViewModels has been effectively replaced. Now we just have to look as data persistence across process death, which can easily be achieved using our old friend onSaveInstanceState
.
As you can see, we have all of the tools provided to us now to control the configChanges in our own application. No more die-recreate Activity lifecycle callbacks when a user simply rotates the phone, or opens the keyboard. Applications can be smooth and responsive, thanks to Compose UI.
Finally, let's talk about handling view state in a ViewModel. Many people like to use Kotlin data classes with val properties to represent their entire view state, and then calling copy() with individual fields changing to handle updates. This is, in fact, how PYDroid itself used to handle state internally. I'm writing today to recommend against this process, for reasons I'll detail now.
Compose runs a re-compose loop as often as it needs to keep things consistent. Let's say you've got a view like this psuedocode:
data class MyViewStateState(val name: String) : UiViewState
var myState = MyViewState(name = "")
fun onNameChanged(name: String) {
myState = myState.copy(name = name)
}
@Composable
fun MyComposable(
state: MyViewState,
onNameChanged: (String) -> Unit,
) {
val name = state.name
TextField(
value = name,
onValueChange = onNameChanged,
)
}
Each time you type a character into the TextField, it will re-compose. This will cause onNameChanged to be called which copies the myState object, allocating more memory, and then stores the variable. It's a ton of allocation and memory, which is slow, and bad.
Let's fix this, by replacing the data class State with an interface
interface MyViewState : UiViewState {
val name: String
}
@ActivityScope
internal class MutableMyViewState @Inject internal constructor() : MyViewState {
override var name by mutableStateOf("")
}
var myState = MutableMyViewState(name = "")
fun onNameChanged(name: String) {
myState.name = name
}
Look at the differences in code now.
Notice how the Composable has not changed because we have correctly separated concerns of our View from our ViewModel, yay!
Now, whenever the name changes, the string will simply be assigned to an already allocated string in the MutableMyViewState, which is private to our ViewModel and to the Composable, MyViewState looks like it has not changed at all because it is now a simple interface. By doing ViewModel state this way we can avoid constantly re-allocating new objects whenever a recomposition happens.
These changes and more will be arriving in PYDroid 24.3.0, which deprecates basically anything touching the old Android View system and makes way for a Compose future.
Stay tuned!
========================
Follow pyamsoft around the Web for updates and announcements about the newest applications!
Like what I do?
Send me an email at: pyam.soft@gmail.com
Or find me online at: https://pyamsoft.blogspot.com
Follow my Facebook Page
Check out my code on GitHub
=========================