|
@@ -1,220 +1,99 @@
|
|
<!DOCTYPE html>
|
|
<!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">
|
|
|
|
|
|
+<html lang="en-us">
|
|
|
|
|
|
|
|
+<head><script src="/livereload.js?mindelay=10&v=2&port=49905&path=livereload" data-no-instant defer></script>
|
|
|
|
+ <title>
|
|
|
|
+MVI Architecture Helper | codeskraps
|
|
|
|
+</title>
|
|
|
|
|
|
|
|
+ <meta http-equiv="content-type" content="text/html; charset=utf-8">
|
|
|
|
+<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
|
|
+<meta name="description" content="Your website description">
|
|
|
|
|
|
|
|
+<meta name="generator" content="Hugo 0.134.3">
|
|
|
|
|
|
-<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)
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-}
|
|
|
|
|
|
+<link rel="canonical" href="http://localhost:49905/posts/mvi_architecture/" >
|
|
|
|
|
|
-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>
|
|
|
|
-}
|
|
|
|
|
|
+<link href="/css/style.min.38d8ec3c7cca8185fe94ab7dc7ca79d69e14b78ca3c077807b2c32725df95c62.css" rel="stylesheet">
|
|
|
|
|
|
-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> {
|
|
|
|
|
|
+</head>
|
|
|
|
|
|
- private val _state = MutableStateFlow(initial)
|
|
|
|
- override val state: StateFlow<STATE> = _state.asStateFlow()
|
|
|
|
|
|
+<body>
|
|
|
|
|
|
- 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="flexWrapper">
|
|
|
|
+ <header class="headerWrapper">
|
|
<div class="header">
|
|
<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>
|
|
|
|
-
|
|
|
|
|
|
+ <div>
|
|
|
|
+ <a class="terminal" href="http://localhost:49905/">
|
|
|
|
+ <span>me@codeskraps.com ~ $</span>
|
|
|
|
+ </a>
|
|
|
|
+ </div>
|
|
|
|
+ <input class="side-menu" type="checkbox" id="side-menu">
|
|
|
|
+ <label class="hamb" for="side-menu"><span class="hamb-line"></span></label>
|
|
|
|
+ <nav class="headerLinks">
|
|
|
|
+ <ul>
|
|
|
|
+
|
|
|
|
+ <li>
|
|
|
|
+ <a href="http://localhost:49905/projects/" title="" >
|
|
|
|
+ ~/projects</a>
|
|
|
|
+ </li>
|
|
|
|
+
|
|
|
|
+ <li>
|
|
|
|
+ <a href="http://localhost:49905/about/" title="" >
|
|
|
|
+ ~/about</a>
|
|
|
|
+ </li>
|
|
|
|
+
|
|
|
|
+ <li>
|
|
|
|
+ <a href="http://localhost:49905/posts/" title="" >
|
|
|
|
+ ~/posts</a>
|
|
|
|
+ </li>
|
|
|
|
+
|
|
|
|
+ </ul>
|
|
</nav>
|
|
</nav>
|
|
</div>
|
|
</div>
|
|
</header>
|
|
</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">
|
|
|
|
|
|
+ <div class="content">
|
|
|
|
+ <main class="main">
|
|
|
|
+
|
|
|
|
+<div class="postWrapper">
|
|
|
|
+ <h1>MVI Architecture Helper</h1>
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ <section class="postMetadata">
|
|
|
|
+ <dl>
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+<dt>tags</dt>
|
|
|
|
+<dd><span></span>
|
|
|
|
+ <a href="/tags/kotlin/">#Kotlin</a><span></span>
|
|
|
|
+ <a href="/tags/kmp/">#Kmp</a><span></span>
|
|
|
|
+ <a href="/tags/android/">#Android</a><span></span>
|
|
|
|
+ <a href="/tags/mvi/">#Mvi</a></dd>
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ <dt>published</dt>
|
|
|
|
+
|
|
|
|
+ <dd><time datetime="2024-09-27">September 27, 2024</time></dd>
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ <dt>reading time</dt>
|
|
|
|
+ <dd>1 minute</dd>
|
|
|
|
+
|
|
|
|
+ </dl>
|
|
|
|
+ </section>
|
|
|
|
+
|
|
|
|
+ <div>
|
|
<p>Something smart to talk about this helper class</p>
|
|
<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> {
|
|
<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">updateState</span>(transform: <span style="color:#66d9ef">suspend</span> (STATE) <span style="color:#f92672">-></span> STATE)
|
|
@@ -286,74 +165,19 @@ class MVIViewModelDelegate<STATE, INTENT, ACTION>(
|
|
</span></span><span style="display:flex;"><span>}</span></span></code></pre></div>
|
|
</span></span><span style="display:flex;"><span>}</span></span></code></pre></div>
|
|
|
|
|
|
</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>
|
|
|
|
|
|
- <div class="side-tags">
|
|
|
|
- <h2>Tags</h2>
|
|
|
|
- <hr />
|
|
|
|
|
|
+ </main>
|
|
|
|
+ </div>
|
|
|
|
|
|
- <ul>
|
|
|
|
-
|
|
|
|
- </ul>
|
|
|
|
-</div>
|
|
|
|
|
|
|
|
- </div>
|
|
|
|
- </main>
|
|
|
|
<footer class="footer">
|
|
<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>
|
|
|
|
-
|
|
|
|
|
|
+
|
|
|
|
+ <span>CC-0, Built with <a href="https://gohugo.io" class="footerLink">Hugo</a> and <a href="https://github.com/LordMathis/hugo-theme-nightfall" class="footerLink">Nightfall</a> theme</span>
|
|
|
|
|
|
</footer>
|
|
</footer>
|
|
|
|
+ </div>
|
|
|
|
+
|
|
|
|
+</body>
|
|
|
|
|
|
- </body>
|
|
|
|
-</html>
|
|
|
|
|
|
+</html>
|