123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 |
- <!DOCTYPE html>
- <html>
- <head><script src="/livereload.js?mindelay=10&v=2&port=1313&path=livereload" data-no-instant defer></script>
- <meta charset="utf-8">
- <meta name="viewport" content="width=device-width,minimum-scale=1">
- <title>MVI Architecture Helper | codeskraps</title>
- <link rel="canonical" href="http://localhost:1313/posts/mvi_architecture/">
- <meta name="description" content="Something smart to talk about this helper class
- interface StateReceiver<STATE> {
- suspend fun updateState(transform: suspend (STATE) -> STATE)
- suspend fun withState(block: suspend (STATE) -> Unit)
- }
- suspend inline fun <reified TYPE : STATE, STATE> StateReceiver<STATE>.withType(crossinline block: suspend (TYPE) -> Unit) {
- withState { state ->
- if (state is TYPE) {
- block(state)
- }
- }
- }
- suspend inline fun <reified TYPE : STATE, STATE> StateReceiver<STATE>.updateWithType(crossinline transform: suspend (TYPE) -> TYPE) {
- withType<TYPE, STATE> { state -> updateState { transform(state) } }
- }
- interface StateProvider<STATE> {
- val state: StateFlow<STATE>
- }
- interface IntentReceiver<INTENT> {
- fun handleIntent(intent: INTENT)
- }
- interface ActionProvider<ACTION> {
- val action: Flow<ACTION>
- }
- interface ActionReceiver<ACTION> {
- suspend fun sendAction(block: suspend () -> ACTION)
- }
- interface StateModule<STATE> : StateReceiver<STATE>, StateProvider<STATE>
- interface IntentModule<INTENT> : IntentReceiver<INTENT>
- interface ActionModule<ACTION> : ActionReceiver<ACTION>, ActionProvider<ACTION>
- interface MVIViewModel<STATE, INTENT, ACTION> :
- StateModule<STATE>,
- IntentModule<INTENT>,
- ActionModule<ACTION>
- class MVIViewModelDelegate<STATE, INTENT, ACTION>(
- initial: STATE
- ) : MVIViewModel<STATE, INTENT, ACTION> {
- private val _state = MutableStateFlow(initial)
- override val state: StateFlow<STATE> = _state.asStateFlow()
- private val _action = Channel<ACTION>()
- override val action: Flow<ACTION> = _action.receiveAsFlow()
- override suspend fun updateState(transform: suspend (STATE) -> STATE) {
- _state.update { transform(it) }
- }
- override suspend fun withState(block: suspend (STATE) -> Unit) {
- block(_state.value)
- }
- override suspend fun sendAction(block: suspend () -> ACTION) {
- _action.trySend(block())
- }
- override fun handleIntent(intent: INTENT) {
- throw NotImplementedError()
- }
- }" />
- <meta property="og:type" content="article" />
- <meta property="og:title" content="MVI Architecture Helper | codeskraps" />
- <meta property="og:url" content="http://localhost:1313/posts/mvi_architecture/" />
- <meta property="og:description" content="Something smart to talk about this helper class
- interface StateReceiver<STATE> {
- suspend fun updateState(transform: suspend (STATE) -> STATE)
- suspend fun withState(block: suspend (STATE) -> Unit)
- }
- suspend inline fun <reified TYPE : STATE, STATE> StateReceiver<STATE>.withType(crossinline block: suspend (TYPE) -> Unit) {
- withState { state ->
- if (state is TYPE) {
- block(state)
- }
- }
- }
- suspend inline fun <reified TYPE : STATE, STATE> StateReceiver<STATE>.updateWithType(crossinline transform: suspend (TYPE) -> TYPE) {
- withType<TYPE, STATE> { state -> updateState { transform(state) } }
- }
- interface StateProvider<STATE> {
- val state: StateFlow<STATE>
- }
- interface IntentReceiver<INTENT> {
- fun handleIntent(intent: INTENT)
- }
- interface ActionProvider<ACTION> {
- val action: Flow<ACTION>
- }
- interface ActionReceiver<ACTION> {
- suspend fun sendAction(block: suspend () -> ACTION)
- }
- interface StateModule<STATE> : StateReceiver<STATE>, StateProvider<STATE>
- interface IntentModule<INTENT> : IntentReceiver<INTENT>
- interface ActionModule<ACTION> : ActionReceiver<ACTION>, ActionProvider<ACTION>
- interface MVIViewModel<STATE, INTENT, ACTION> :
- StateModule<STATE>,
- IntentModule<INTENT>,
- ActionModule<ACTION>
- class MVIViewModelDelegate<STATE, INTENT, ACTION>(
- initial: STATE
- ) : MVIViewModel<STATE, INTENT, ACTION> {
- private val _state = MutableStateFlow(initial)
- override val state: StateFlow<STATE> = _state.asStateFlow()
- private val _action = Channel<ACTION>()
- override val action: Flow<ACTION> = _action.receiveAsFlow()
- override suspend fun updateState(transform: suspend (STATE) -> STATE) {
- _state.update { transform(it) }
- }
- override suspend fun withState(block: suspend (STATE) -> Unit) {
- block(_state.value)
- }
- override suspend fun sendAction(block: suspend () -> ACTION) {
- _action.trySend(block())
- }
- override fun handleIntent(intent: INTENT) {
- throw NotImplementedError()
- }
- }" />
- <link rel="stylesheet" href="/lib/icofont/icofont.min.css" />
- <link rel="stylesheet" href="/css/syntax.css" />
- <link rel="stylesheet" href="/css/style.css" />
- <script src="/js/copy-code-block.js"></script>
- <link rel="shortcut icon" href="/images/favicon.ico" type="image/x-icon" />
- </head>
- <body>
- <header class="header-wrapper">
- <div class="header">
- <a class="site-title" href="http://localhost:1313/">codeskraps</a>
- <nav class="menu">
-
- <div class="menu-item">
-
- <a href="/about/">About</a>
-
- </div>
-
- <div class="menu-item">
-
- <a href="/posts/">Posts</a>
-
- </div>
-
- </nav>
- </div>
- </header>
- <main class="main-wrapper">
- <div class="main">
-
- <section class="single">
- <h1 class="title">MVI Architecture Helper</h1>
- <div class="tip">
- <time datetime="2024-09-27 13:45:09 +0200 CEST">2024/09/27</time>
- <span class="split">·</span>
- <span> 213 words </span>
- <span class="split">·</span>
- <span>
- 1 minutes to read
- </span>
- </div>
- <div class="taxonomies">
-
-
- </div>
- <hr />
- <div class="content">
- <p>Something smart to talk about this helper class</p>
- <div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;"><code class="language-kotlin" data-lang="kotlin"><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">StateReceiver</span><STATE> {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">updateState</span>(transform: <span style="color:#66d9ef">suspend</span> (STATE) <span style="color:#f92672">-></span> STATE)
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">withState</span>(block: <span style="color:#66d9ef">suspend</span> (STATE) <span style="color:#f92672">-></span> Unit)
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">inline</span> <span style="color:#66d9ef">fun</span> <<span style="color:#66d9ef">reified</span> <span style="color:#a6e22e">TYPE</span> : <span style="color:#a6e22e">STATE</span>, <span style="color:#a6e22e">STATE</span>> <span style="color:#a6e22e">StateReceiver</span><STATE>.withType(<span style="color:#66d9ef">crossinline</span> block: <span style="color:#66d9ef">suspend</span> (TYPE) <span style="color:#f92672">-></span> Unit) {
- </span></span><span style="display:flex;"><span> withState { state <span style="color:#f92672">-></span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">if</span> (state <span style="color:#66d9ef">is</span> TYPE) {
- </span></span><span style="display:flex;"><span> block(state)
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">inline</span> <span style="color:#66d9ef">fun</span> <<span style="color:#66d9ef">reified</span> <span style="color:#a6e22e">TYPE</span> : <span style="color:#a6e22e">STATE</span>, <span style="color:#a6e22e">STATE</span>> <span style="color:#a6e22e">StateReceiver</span><STATE>.updateWithType(<span style="color:#66d9ef">crossinline</span> transform: <span style="color:#66d9ef">suspend</span> (TYPE) <span style="color:#f92672">-></span> TYPE) {
- </span></span><span style="display:flex;"><span> withType<TYPE, STATE> { state <span style="color:#f92672">-></span> updateState { transform(state) } }
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">StateProvider</span><STATE> {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">val</span> state: StateFlow<STATE>
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">IntentReceiver</span><INTENT> {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">handleIntent</span>(intent: INTENT)
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">ActionProvider</span><ACTION> {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">val</span> action: Flow<ACTION>
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">ActionReceiver</span><ACTION> {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">sendAction</span>(block: <span style="color:#66d9ef">suspend</span> () <span style="color:#f92672">-></span> ACTION)
- </span></span><span style="display:flex;"><span>}
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">StateModule</span><STATE> : StateReceiver<STATE>, StateProvider<STATE>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">IntentModule</span><INTENT> : IntentReceiver<INTENT>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">ActionModule</span><ACTION> : ActionReceiver<ACTION>, ActionProvider<ACTION>
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">MVIViewModel</span><STATE, INTENT, ACTION> :
- </span></span><span style="display:flex;"><span> StateModule<STATE>,
- </span></span><span style="display:flex;"><span> IntentModule<INTENT>,
- </span></span><span style="display:flex;"><span> ActionModule<ACTION>
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span><span style="color:#66d9ef">class</span> <span style="color:#a6e22e">MVIViewModelDelegate</span><STATE, INTENT, ACTION>(
- </span></span><span style="display:flex;"><span> initial: STATE
- </span></span><span style="display:flex;"><span>) : MVIViewModel<STATE, INTENT, ACTION> {
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">private</span> <span style="color:#66d9ef">val</span> _state = MutableStateFlow(initial)
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">val</span> state: StateFlow<STATE> = _state.asStateFlow()
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">private</span> <span style="color:#66d9ef">val</span> _action = Channel<ACTION>()
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">val</span> action: Flow<ACTION> = _action.receiveAsFlow()
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">updateState</span>(transform: <span style="color:#66d9ef">suspend</span> (STATE) <span style="color:#f92672">-></span> STATE) {
- </span></span><span style="display:flex;"><span> _state.update { transform(<span style="color:#66d9ef">it</span>) }
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">withState</span>(block: <span style="color:#66d9ef">suspend</span> (STATE) <span style="color:#f92672">-></span> Unit) {
- </span></span><span style="display:flex;"><span> block(_state.<span style="color:#66d9ef">value</span>)
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">suspend</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">sendAction</span>(block: <span style="color:#66d9ef">suspend</span> () <span style="color:#f92672">-></span> ACTION) {
- </span></span><span style="display:flex;"><span> _action.trySend(block())
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span>
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">override</span> <span style="color:#66d9ef">fun</span> <span style="color:#a6e22e">handleIntent</span>(intent: INTENT) {
- </span></span><span style="display:flex;"><span> <span style="color:#66d9ef">throw</span> NotImplementedError()
- </span></span><span style="display:flex;"><span> }
- </span></span><span style="display:flex;"><span>}</span></span></code></pre></div>
- </div>
-
- </section>
- </div>
- <div class="side">
-
- <div class="side-recent">
- <h2 class="side-title">
- <a href="/posts/">Recent Posts</a>
- </h2>
- <hr />
- <ul>
-
- <li>
- <a href="/posts/mvi_architecture/">MVI Architecture Helper</a>
- </li>
-
- <li>
- <a href="/posts/my-first-post/">My First Post</a>
- </li>
-
- </ul>
- </div>
-
- <div class="side-categories">
- <h2>Categories</h2>
- <hr />
- <ul>
-
- </ul>
- </div>
- <div class="side-tags">
- <h2>Tags</h2>
- <hr />
- <ul>
-
- </ul>
- </div>
- </div>
- </main>
- <footer class="footer">
- <div class="footer-row">
-
-
-
- <a class="footer-item" href="http://localhost:1313/posts/index.xml">
- Feed of Posts
- <i class="icofont-rss"></i>
- </a>
-
-
-
-
-
-
- </div>
-
- </footer>
- </body>
- </html>
|