From 9a5d94dbf8ca8a45dd9f244d6464fe4b70328231 Mon Sep 17 00:00:00 2001 From: Laimonas Turauskas Date: Mon, 11 Dec 2023 14:49:51 -0800 Subject: [PATCH] [Fragment] Replace bind with setOutput function. --- .../android/compose/ComposeViewFactory.kt | 34 +++-------- .../instacart/formula/android/FeatureView.kt | 13 +--- .../formula/android/FormulaFragment.kt | 60 +++++++++++++------ .../instacart/formula/android/ViewInstance.kt | 3 +- .../android/views/FeatureViewBindFunction.kt | 42 ------------- 5 files changed, 55 insertions(+), 97 deletions(-) delete mode 100644 formula-android/src/main/java/com/instacart/formula/android/views/FeatureViewBindFunction.kt diff --git a/formula-android-compose/src/main/java/com/instacart/formula/android/compose/ComposeViewFactory.kt b/formula-android-compose/src/main/java/com/instacart/formula/android/compose/ComposeViewFactory.kt index e32a45734..6c5b76786 100644 --- a/formula-android-compose/src/main/java/com/instacart/formula/android/compose/ComposeViewFactory.kt +++ b/formula-android-compose/src/main/java/com/instacart/formula/android/compose/ComposeViewFactory.kt @@ -1,6 +1,5 @@ package com.instacart.formula.android.compose -import android.os.SystemClock import android.view.LayoutInflater import android.view.ViewGroup import androidx.compose.runtime.Composable @@ -8,38 +7,23 @@ import androidx.compose.runtime.rxjava3.subscribeAsState import androidx.compose.ui.platform.ComposeView import com.instacart.formula.android.FeatureView import com.instacart.formula.android.ViewFactory +import com.jakewharton.rxrelay3.BehaviorRelay abstract class ComposeViewFactory : ViewFactory { override fun create(inflater: LayoutInflater, container: ViewGroup?): FeatureView { val view = ComposeView(inflater.context) - var firstRender = true + val outputRelay = BehaviorRelay.create() + view.setContent { + val model = outputRelay.subscribeAsState(null).value + if (model != null) { + Content(model) + } + } return FeatureView( view = view, - bind = { state -> - view.setContent { - val model = state.observable.subscribeAsState(null).value - if (model != null) { - val start = SystemClock.uptimeMillis() - Content(model) - val end = SystemClock.uptimeMillis() - state.environment.eventListener?.onRendered( - fragmentId = state.fragmentId, - durationInMillis = end - start, - ) - - if (firstRender) { - firstRender = false - state.environment.eventListener?.onFirstModelRendered( - fragmentId = state.fragmentId, - durationInMillis = end - state.initializedAtMillis, - ) - } - } - } - null - } + setOutput = outputRelay::accept, ) } diff --git a/formula-android/src/main/java/com/instacart/formula/android/FeatureView.kt b/formula-android/src/main/java/com/instacart/formula/android/FeatureView.kt index f4b24a53a..55d4ad977 100644 --- a/formula-android/src/main/java/com/instacart/formula/android/FeatureView.kt +++ b/formula-android/src/main/java/com/instacart/formula/android/FeatureView.kt @@ -13,18 +13,11 @@ import io.reactivex.rxjava3.core.Observable * [FeatureView]. * * @param view The root Android view. - * @param bind A bind function connects state observable to the view rendering. + * @param setOutput A function called to apply [RenderModel] to the view. * @param lifecycleCallbacks Optional lifecycle callbacks if you need to know the Fragment state. */ class FeatureView( val view: View, - val bind: (State) -> Cancelable?, + val setOutput: (RenderModel) -> Unit, val lifecycleCallbacks: FragmentLifecycleCallback? = null, -) { - class State( - val initializedAtMillis: Long, - val fragmentId: FragmentId, - val environment: FragmentEnvironment, - val observable: Observable, - ) -} \ No newline at end of file +) diff --git a/formula-android/src/main/java/com/instacart/formula/android/FormulaFragment.kt b/formula-android/src/main/java/com/instacart/formula/android/FormulaFragment.kt index 1ea8f23af..565aef0d8 100644 --- a/formula-android/src/main/java/com/instacart/formula/android/FormulaFragment.kt +++ b/formula-android/src/main/java/com/instacart/formula/android/FormulaFragment.kt @@ -11,6 +11,7 @@ import com.instacart.formula.Cancelable import com.instacart.formula.android.internal.FormulaFragmentDelegate import com.instacart.formula.android.internal.getFormulaFragmentId import com.jakewharton.rxrelay3.BehaviorRelay +import java.lang.Exception class FormulaFragment : Fragment(), BaseFormulaFragment { companion object { @@ -33,9 +34,9 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { private var initializedAtMillis: Long? = SystemClock.uptimeMillis() + private var firstRender = true + private var output: Any? = null private var featureView: FeatureView? = null - private val stateRelay: BehaviorRelay = BehaviorRelay.create() - private var cancelable: Cancelable? = null private var lifecycleCallback: FragmentLifecycleCallback? = null @@ -47,6 +48,8 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { } override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { + firstRender = true + val viewFactory = FormulaFragmentDelegate.viewFactory(this) ?: run { // No view factory, no view return null @@ -59,17 +62,10 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) - featureView?.let { value -> - val state = FeatureView.State( - initializedAtMillis = initializedAtMillis ?: SystemClock.uptimeMillis(), - fragmentId = getFormulaFragmentId(), - environment = FormulaFragmentDelegate.fragmentEnvironment(), - observable = stateRelay, - ) - cancelable = value.bind(state) - this.lifecycleCallback = value.lifecycleCallbacks - lifecycleCallback?.onViewCreated(view, savedInstanceState) - } + tryToSetState() + + lifecycleCallback = featureView?.lifecycleCallbacks + lifecycleCallback?.onViewCreated(view, savedInstanceState) } override fun onActivityCreated(savedInstanceState: Bundle?) { @@ -110,9 +106,6 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { override fun onDestroyView() { initializedAtMillis = null - cancelable?.cancel() - cancelable = null - lifecycleCallback?.onDestroyView() lifecycleCallback = null super.onDestroyView() @@ -120,11 +113,12 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { } override fun setState(state: Any) { - stateRelay.accept(state) + output = state + tryToSetState() } override fun currentState(): Any? { - return stateRelay.value + return output } override fun getFragmentKey(): FragmentKey { @@ -134,4 +128,34 @@ class FormulaFragment : Fragment(), BaseFormulaFragment { override fun toString(): String { return "${key.tag} -> $key" } + + private fun tryToSetState() { + val output = output ?: return + val view = featureView ?: return + + val fragmentId = getFormulaFragmentId() + val environment = FormulaFragmentDelegate.fragmentEnvironment() + + try { + val start = SystemClock.uptimeMillis() + view.setOutput(output) + val end = SystemClock.uptimeMillis() + + + environment.eventListener?.onRendered( + fragmentId = fragmentId, + durationInMillis = end - start, + ) + + if (firstRender) { + firstRender = false + environment.eventListener?.onFirstModelRendered( + fragmentId = fragmentId, + durationInMillis = end - (initializedAtMillis ?: SystemClock.uptimeMillis()), + ) + } + } catch (exception: Exception) { + environment.onScreenError(key, exception) + } + } } diff --git a/formula-android/src/main/java/com/instacart/formula/android/ViewInstance.kt b/formula-android/src/main/java/com/instacart/formula/android/ViewInstance.kt index 2123b9393..aafb9b1a4 100644 --- a/formula-android/src/main/java/com/instacart/formula/android/ViewInstance.kt +++ b/formula-android/src/main/java/com/instacart/formula/android/ViewInstance.kt @@ -3,7 +3,6 @@ package com.instacart.formula.android import android.view.View import com.instacart.formula.RenderView import com.instacart.formula.Renderer -import com.instacart.formula.android.views.FeatureViewBindFunction /** * View instance contains an initialized Android [view] and provides factory @@ -37,7 +36,7 @@ abstract class ViewInstance { ): FeatureView { return FeatureView( view = view, - bind = FeatureViewBindFunction(renderer), + setOutput = renderer, lifecycleCallbacks = lifecycleCallbacks ) } diff --git a/formula-android/src/main/java/com/instacart/formula/android/views/FeatureViewBindFunction.kt b/formula-android/src/main/java/com/instacart/formula/android/views/FeatureViewBindFunction.kt deleted file mode 100644 index 56f083e52..000000000 --- a/formula-android/src/main/java/com/instacart/formula/android/views/FeatureViewBindFunction.kt +++ /dev/null @@ -1,42 +0,0 @@ -package com.instacart.formula.android.views - -import android.os.SystemClock -import com.instacart.formula.Cancelable -import com.instacart.formula.Renderer -import com.instacart.formula.android.FeatureView -import java.lang.Exception - -/** - * Binds [FeatureView.State] to a [render] function. - */ -internal class FeatureViewBindFunction( - private val render: Renderer -) : (FeatureView.State) -> Cancelable? { - override fun invoke(state: FeatureView.State): Cancelable { - val environment = state.environment - var firstRender = true - val disposable = state.observable.subscribe { - try { - val start = SystemClock.uptimeMillis() - render(it) - val end = SystemClock.uptimeMillis() - environment.eventListener?.onRendered( - fragmentId = state.fragmentId, - durationInMillis = end - start, - ) - if (firstRender) { - firstRender = false - environment.eventListener?.onFirstModelRendered( - fragmentId = state.fragmentId, - durationInMillis = end - state.initializedAtMillis, - ) - } - } catch (exception: Exception) { - environment.onScreenError(state.fragmentId.key, exception) - } - } - return Cancelable { - disposable.dispose() - } - } -} \ No newline at end of file