Skip to content

Commit

Permalink
Return the Job from WorkflowLayout.take so it can be canceled.
Browse files Browse the repository at this point in the history
  • Loading branch information
rjrjr committed Aug 7, 2024
1 parent 7316b0d commit a90bf43
Show file tree
Hide file tree
Showing 5 changed files with 70 additions and 7 deletions.
4 changes: 2 additions & 2 deletions workflow-ui/core-android/api/core-android.api
Original file line number Diff line number Diff line change
Expand Up @@ -128,8 +128,8 @@ public final class com/squareup/workflow1/ui/WorkflowLayout : android/widget/Fra
public synthetic fun <init> (Landroid/content/Context;Landroid/util/AttributeSet;ILkotlin/jvm/internal/DefaultConstructorMarker;)V
public final fun show (Lcom/squareup/workflow1/ui/Screen;Lcom/squareup/workflow1/ui/ViewEnvironment;)V
public static synthetic fun show$default (Lcom/squareup/workflow1/ui/WorkflowLayout;Lcom/squareup/workflow1/ui/Screen;Lcom/squareup/workflow1/ui/ViewEnvironment;ILjava/lang/Object;)V
public final fun take (Landroidx/lifecycle/Lifecycle;Lkotlinx/coroutines/flow/Flow;Landroidx/lifecycle/Lifecycle$State;Lkotlin/coroutines/CoroutineContext;)V
public static synthetic fun take$default (Lcom/squareup/workflow1/ui/WorkflowLayout;Landroidx/lifecycle/Lifecycle;Lkotlinx/coroutines/flow/Flow;Landroidx/lifecycle/Lifecycle$State;Lkotlin/coroutines/CoroutineContext;ILjava/lang/Object;)V
public final fun take (Landroidx/lifecycle/Lifecycle;Lkotlinx/coroutines/flow/Flow;Landroidx/lifecycle/Lifecycle$State;Lkotlin/coroutines/CoroutineContext;)Lkotlinx/coroutines/Job;
public static synthetic fun take$default (Lcom/squareup/workflow1/ui/WorkflowLayout;Landroidx/lifecycle/Lifecycle;Lkotlinx/coroutines/flow/Flow;Landroidx/lifecycle/Lifecycle$State;Lkotlin/coroutines/CoroutineContext;ILjava/lang/Object;)Lkotlinx/coroutines/Job;
}

public final class com/squareup/workflow1/ui/WorkflowViewStub : android/view/View {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import androidx.lifecycle.repeatOnLifecycle
import com.squareup.workflow1.ui.androidx.OnBackPressedDispatcherOwnerKey
import com.squareup.workflow1.ui.androidx.WorkflowAndroidXSupport.onBackPressedDispatcherOwnerOrNull
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.Job
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.launch
import kotlin.coroutines.CoroutineContext
Expand Down Expand Up @@ -91,19 +92,23 @@ public class WorkflowLayout(
* @param [collectionContext] additional [CoroutineContext] we want for the coroutine that is
* launched to collect the renderings. This should not override the [CoroutineDispatcher][kotlinx.coroutines.CoroutineDispatcher]
* but may include some other instrumentation elements.
*
* @return the [Job] started to collect [renderings], to allow callers to
* [cancel][Job.cancel] collection -- e.g., before calling [take] again with a new
* [renderings] flow
*/
@OptIn(ExperimentalStdlibApi::class)
public fun take(
lifecycle: Lifecycle,
renderings: Flow<Screen>,
repeatOnLifecycle: State = STARTED,
collectionContext: CoroutineContext = EmptyCoroutineContext
) {
): Job {
// We remove the dispatcher as we want to use what is provided by the lifecycle.coroutineScope.
val contextWithoutDispatcher = collectionContext.minusKey(CoroutineDispatcher.Key)
val lifecycleDispatcher = lifecycle.coroutineScope.coroutineContext[CoroutineDispatcher.Key]
// Just like https://medium.com/androiddevelopers/a-safer-way-to-collect-flows-from-android-uis-23080b1f8bda
lifecycle.coroutineScope.launch(contextWithoutDispatcher) {
return lifecycle.coroutineScope.launch(contextWithoutDispatcher) {
lifecycle.repeatOnLifecycle(repeatOnLifecycle) {
require(coroutineContext[CoroutineDispatcher.Key] == lifecycleDispatcher) {
"Collection dispatch should happen on the lifecycle's dispatcher."
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -230,7 +230,15 @@ public class WorkflowViewStub @JvmOverloads constructor(

holder = rendering.toViewFactory(viewEnvironment)
.startShowing(rendering, viewEnvironment, parent.context, parent) { view, doStart ->
WorkflowLifecycleOwner.installOn(view, viewEnvironment.onBackPressedDispatcherOwner(parent))
try {
WorkflowLifecycleOwner.installOn(
view,
viewEnvironment.onBackPressedDispatcherOwner(parent)
)
} catch ( t: Throwable) {
println(t.toString())
throw t
}
doStart()
}.apply {
val newView = view
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,8 @@ public interface WorkflowLifecycleOwner : LifecycleOwner {
onBackPressedDispatcherOwner: OnBackPressedDispatcherOwner,
findParentLifecycle: (View) -> Lifecycle = this::findParentViewTreeLifecycle
) {
RealWorkflowLifecycleOwner(findParentLifecycle).also {
val wlo = RealWorkflowLifecycleOwner(findParentLifecycle)
wlo.also {
view.setViewTreeOnBackPressedDispatcherOwner(onBackPressedDispatcherOwner)
view.setViewTreeLifecycleOwner(it)
view.addOnAttachStateChangeListener(it)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,16 +5,21 @@ import android.os.Bundle
import android.os.Parcelable
import android.util.SparseArray
import android.view.View
import androidx.activity.OnBackPressedDispatcher
import androidx.activity.OnBackPressedDispatcherOwner
import androidx.activity.setViewTreeOnBackPressedDispatcherOwner
import androidx.core.view.get
import androidx.lifecycle.Lifecycle
import androidx.lifecycle.testing.TestLifecycleOwner
import androidx.test.core.app.ApplicationProvider
import com.google.common.truth.Truth.assertThat
import com.squareup.workflow1.ui.androidx.OnBackPressedDispatcherOwnerKey
import com.squareup.workflow1.ui.navigation.WrappedScreen
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.flowOf
import kotlinx.coroutines.test.UnconfinedTestDispatcher
import kotlinx.coroutines.test.runTest
import org.junit.Test
import org.junit.runner.RunWith
import org.robolectric.RobolectricTestRunner
Expand All @@ -28,7 +33,13 @@ import kotlin.coroutines.CoroutineContext
internal class WorkflowLayoutTest {
private val context: Context = ApplicationProvider.getApplicationContext()

private val workflowLayout = WorkflowLayout(context).apply { id = 42 }
private val workflowLayout = WorkflowLayout(context).apply {
id = 42
setViewTreeOnBackPressedDispatcherOwner(object : OnBackPressedDispatcherOwner {
override fun getOnBackPressedDispatcher(): OnBackPressedDispatcher { error("yeah no") }
override val lifecycle: Lifecycle get() = error("nope")
})
}

@Test fun ignoresAlienViewState() {
val weirdView = BundleSavingView(context)
Expand Down Expand Up @@ -91,6 +102,44 @@ internal class WorkflowLayoutTest {
// No crash then we safely removed the dispatcher.
}

@Test fun takes() {
val lifecycleDispatcher = UnconfinedTestDispatcher()
val testLifecycle = TestLifecycleOwner(
initialState = Lifecycle.State.RESUMED,
coroutineDispatcher = lifecycleDispatcher
)
val flow = MutableSharedFlow<Screen>()

runTest(lifecycleDispatcher) {
val job = workflowLayout.take(
lifecycle = testLifecycle.lifecycle,
renderings = flow,
)
assertThat(workflowLayout[0]).isInstanceOf(WorkflowViewStub::class.java)
flow.emit(WrappedScreen())
assertThat(workflowLayout[0]).isNotInstanceOf(WorkflowViewStub::class.java)
}
}

@Test fun canStopTaking() {
val lifecycleDispatcher = UnconfinedTestDispatcher()
val testLifecycle = TestLifecycleOwner(
initialState = Lifecycle.State.RESUMED,
coroutineDispatcher = lifecycleDispatcher
)
val flow = MutableSharedFlow<Screen>()

runTest(lifecycleDispatcher) {
val job = workflowLayout.take(
lifecycle = testLifecycle.lifecycle,
renderings = flow,
)
job.cancel()
flow.emit(WrappedScreen())
assertThat(workflowLayout[0]).isInstanceOf(WorkflowViewStub::class.java)
}
}

private class BundleSavingView(context: Context) : View(context) {
var saved = false

Expand Down

0 comments on commit a90bf43

Please sign in to comment.