diff --git a/src/commonMain/kotlin/dev/mcatta/polpetta/ReducerFactory.kt b/src/commonMain/kotlin/dev/mcatta/polpetta/ReducerFactory.kt index 2dba669..088748e 100644 --- a/src/commonMain/kotlin/dev/mcatta/polpetta/ReducerFactory.kt +++ b/src/commonMain/kotlin/dev/mcatta/polpetta/ReducerFactory.kt @@ -16,8 +16,7 @@ public abstract class ReducerFactory( sideEffectFactory: SideEffectFactory ) { - private val _reducerDefinition: MutableMap, MutableList>> = - mutableMapOf() + private val _reducerDefinition: MutableList> = mutableListOf() private val _sideEffectFactory = sideEffectFactory /** @@ -30,10 +29,9 @@ public abstract class ReducerFactory( internal fun getReducer( action: A, fromState: FromState - ): Reducer? = - _reducerDefinition[fromState::class] - ?.firstOrNull { item -> item.kClassAction == action::class } - ?.build(action) + ): Reducer? = _reducerDefinition + .firstOrNull { item -> item.kClassAction == action::class && item.kClassFromState.isInstance(fromState) } + ?.build(action) /** * Define a [Reducer]'s body for the defined action [A] @@ -58,9 +56,10 @@ public abstract class ReducerFactory( kClassFromState: KClass, block: suspend SideEffectFactory.(RA, StateModifier) -> S ) { - _reducerDefinition.getOrPut(kClassFromState) { mutableListOf() }.add( + _reducerDefinition.add( ReducerFactoryBuilder( kClassAction = kClassAction, + kClassFromState = kClassFromState, handler = { action -> @Suppress("UNCHECKED_CAST") (reducer { state -> block(_sideEffectFactory, action as RA, state as StateModifier) }) diff --git a/src/commonMain/kotlin/dev/mcatta/polpetta/core/ReducerFactoryBuilder.kt b/src/commonMain/kotlin/dev/mcatta/polpetta/core/ReducerFactoryBuilder.kt index 780acb2..e4bd394 100644 --- a/src/commonMain/kotlin/dev/mcatta/polpetta/core/ReducerFactoryBuilder.kt +++ b/src/commonMain/kotlin/dev/mcatta/polpetta/core/ReducerFactoryBuilder.kt @@ -8,10 +8,12 @@ import kotlin.reflect.KClass * Reducer Factory Builder used to wrap the Reduce function based on an action * * @param kClassAction action type + * @param kClassFromState input state type * @param handler reducer */ internal class ReducerFactoryBuilder( val kClassAction: KClass, + val kClassFromState: KClass, val handler: (A) -> Reducer ) { fun build(action: A): Reducer = handler.invoke(action) diff --git a/src/commonTest/kotlin/dev/mcatta/polpetta/StateStoreTest.kt b/src/commonTest/kotlin/dev/mcatta/polpetta/StateStoreTest.kt index eb781d0..af7bd4e 100644 --- a/src/commonTest/kotlin/dev/mcatta/polpetta/StateStoreTest.kt +++ b/src/commonTest/kotlin/dev/mcatta/polpetta/StateStoreTest.kt @@ -99,6 +99,35 @@ internal class StateStoreTest { } } + @Test + fun `Test action based on a Generic State`() = runTest(context = testScope.coroutineContext) { + // Given + val testStore = object : StateStore( + coroutineScope = testScope, + initialState = TestState.Count(0), + reducerFactory = { + on { _, stateModifier -> + stateModifier.transform { copy(counter = counter + 1) } + } + + on { _, stateModifier -> + stateModifier.transform { TestState.Result(this.toString()) } + } + } + ) {} + + testStore.stateFlow.test { + // When + testStore.dispatchAction(TestAction.Increase) + testStore.dispatchAction(TestAction.ToString) + + // Then + assertIs(awaitItem()) + assertIs(awaitItem()) + assertIs(awaitItem()) + } + } + @Test fun `Test Intents execution and side effect`() = runTest(context = testScope.coroutineContext) { // Given