Skip to content

Commit

Permalink
Introduces SavedStateHandle.removeWorkflowState()
Browse files Browse the repository at this point in the history
If you're juggling multiple workflow runtimes per `Activity` (if you're the kind of person who needs to call `Job.cancel` on the `Job` returned from `WorkflowLayout.take`) then you might also need to throw away the state captured by an AndroidX `SavedStateHandler` passed to the Android flavor of `renderWorkflowIn()` to prevent memory leaks.
  • Loading branch information
rjrjr committed Nov 5, 2024
1 parent b91ad6c commit c905ad1
Show file tree
Hide file tree
Showing 4 changed files with 107 additions and 1 deletion.
1 change: 1 addition & 0 deletions workflow-ui/core-android/api/core-android.api
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
public final class com/squareup/workflow1/ui/AndroidRenderWorkflowKt {
public static final fun removeWorkflowState (Landroidx/lifecycle/SavedStateHandle;)V
public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow;
public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Ljava/lang/Object;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow;
public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow;
Expand Down
3 changes: 3 additions & 0 deletions workflow-ui/core-android/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,10 @@ android {
}

dependencies {
androidTestImplementation(libs.androidx.activity.ktx)
androidTestImplementation(libs.androidx.appcompat)
androidTestImplementation(libs.androidx.lifecycle.viewmodel.ktx)
androidTestImplementation(libs.androidx.lifecycle.viewmodel.savedstate)
androidTestImplementation(libs.truth)

api(libs.androidx.lifecycle.common)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
package com.squareup.workflow1.ui

import android.widget.FrameLayout
import androidx.activity.ComponentActivity
import androidx.activity.viewModels
import androidx.lifecycle.Lifecycle.State.CREATED
import androidx.lifecycle.SavedStateHandle
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import androidx.test.ext.junit.rules.ActivityScenarioRule
import com.google.common.truth.Truth.assertThat
import com.squareup.workflow1.StatelessWorkflow
import com.squareup.workflow1.ui.internal.test.IdlingDispatcherRule
import kotlinx.coroutines.Job
import kotlinx.coroutines.flow.StateFlow
import leakcanary.DetectLeaksAfterTestSuccess
import org.junit.Rule
import org.junit.Test
import org.junit.rules.RuleChain

@OptIn(WorkflowUiExperimentalApi::class)
internal class AndroidRenderWorkflowInTest {
@get:Rule val scenarioRule = ActivityScenarioRule(ComponentActivity::class.java)
private val scenario get() = scenarioRule.scenario

@get:Rule val rules: RuleChain = RuleChain.outerRule(DetectLeaksAfterTestSuccess())
.around(scenarioRule)
.around(IdlingDispatcherRule)

@Test fun removeWorkflowStateDoesWhatItSaysOnTheTin() {
var job: Job? = null

// Activity.onCreate(), the take() call won't start pulling yet.
scenario.onActivity { activity ->
val model: SomeViewModel by activity.viewModels()
val renderings: StateFlow<Screen> = renderWorkflowIn(
workflow = SomeWorkflow,
scope = model.viewModelScope,
savedStateHandle = model.savedStateHandle
)

val layout = WorkflowLayout(activity)
activity.setContentView(layout)

assertThat(model.savedStateHandle.contains(KEY)).isFalse()

job = layout.take(activity.lifecycle, renderings)
assertThat(model.savedStateHandle.contains(KEY)).isFalse()
}

// Exit onCreate() and move to CREATED status. take() starts to draw
// and the renderWorkflowIn() call above starts pushing TreeSnapshots
// (lazy serialization functions) to the SavedStateHandle.
scenario.moveToState(CREATED)
scenario.onActivity { activity ->
val model: SomeViewModel by activity.viewModels()
assertThat(model.savedStateHandle.contains(KEY)).isTrue()

// The Job returned from take() is canceled. There is still a
// TreeSnapshot and whatever pointer it captured in the SavedStateHandle.
job?.cancel()
assertThat(model.savedStateHandle.contains(KEY)).isTrue()

// We can remove it.
model.savedStateHandle.removeWorkflowState()
assertThat(model.savedStateHandle.contains(KEY)).isFalse()
}
}

object SomeScreen : AndroidScreen<SomeScreen> {
override val viewFactory: ScreenViewFactory<SomeScreen> =
ScreenViewFactory.fromCode { _, initialEnvironment, context, _ ->
ScreenViewHolder(
initialEnvironment,
FrameLayout(context)
) { _, _ -> }
}
}

object SomeWorkflow : StatelessWorkflow<Unit, Nothing, Screen>() {
override fun render(
renderProps: Unit,
context: RenderContext
): Screen {
return SomeScreen
}
}

class SomeViewModel(val savedStateHandle: SavedStateHandle) : ViewModel()
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package com.squareup.workflow1.ui

import androidx.annotation.VisibleForTesting
import androidx.lifecycle.SavedStateHandle
import com.squareup.workflow1.RuntimeConfig
import com.squareup.workflow1.RuntimeConfigOptions
Expand Down Expand Up @@ -290,4 +291,15 @@ public fun <PropsT, OutputT, RenderingT> renderWorkflowIn(
.stateIn(scope, Eagerly, renderingsAndSnapshots.value.rendering)
}

private const val KEY = "com.squareup.workflow1.ui.renderWorkflowIn-snapshot"
/**
* Removes state added to the `savedStateHandle` argument of the Android-specific
* overload of [renderWorkflowIn]. For use in obscure cases like swapping between
* different Workflow runtimes in an app. Most apps will not use this function.
*/
@WorkflowUiExperimentalApi
public fun SavedStateHandle.removeWorkflowState() {
remove<Any>(KEY)
}

@VisibleForTesting
internal const val KEY = "com.squareup.workflow1.ui.renderWorkflowIn-snapshot"

0 comments on commit c905ad1

Please sign in to comment.