This repository was archived by the owner on Sep 30, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathExampleComponent.kt
107 lines (92 loc) · 3.65 KB
/
ExampleComponent.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
package y2k.rxstateexample.react
import kotlinx.coroutines.experimental.Job
import kotlinx.coroutines.experimental.android.UI
import kotlinx.coroutines.experimental.channels.BroadcastChannel
import kotlinx.coroutines.experimental.launch
import y2k.rxstateexample.common.Service
import y2k.rxstateexample.common.toUser
import y2k.rxstateexample.react.ExampleComponent.Events
import y2k.rxstateexample.react.ExampleComponent.State
class ExampleComponent(private val service: Service) : Component<State, Events>() {
/** Обработка входящих событий. */
override fun update(event: Events) {
when (event) {
is Events.EmailChanged -> updateState(getState().copy(email = event.value))
is Events.NameChanged -> updateState(getState().copy(name = event.value))
is Events.SurnameChanged -> updateState(getState().copy(surname = event.value))
Events.Clicked -> {
service.saveUser(getState().toUser())
updateState(getState())
}
}
}
private fun updateState(state: State) {
val emailValid = state.email.contains("@")
val usersEqual = service.readUser() == state.toUser()
val newState = state.copy(
emailValid = emailValid,
buttonEnabled = emailValid && !usersEqual,
buttonText = if (usersEqual) "Nothing changed" else "Save changes")
setState(newState)
}
/** Создание начального стейта. */
override fun getInitState(): State = State(buttonText = "Nothing changed")
/** Весь "важный" для экрана UI стейт. */
data class State(
val email: String = "",
val name: String = "",
val surname: String = "",
val buttonText: String = "",
val buttonEnabled: Boolean = false,
val emailValid: Boolean = true)
/** События приходящие от UI в компонент. */
sealed class Events {
class EmailChanged(val value: String) : Events()
class NameChanged(val value: String) : Events()
class SurnameChanged(val value: String) : Events()
object Clicked : Events()
}
}
// *** *** *** *** *** *** *** *** *** *** *** ***
// Мини фреймворк
// *** *** *** *** *** *** *** *** *** *** *** ***
abstract class Component<TState, TEvent> {
private var state: TState? = null
private var callback: ((TState) -> Unit)? = null
private var subscriptionJob: Job? = null
abstract fun getInitState(): TState
abstract fun update(event: TEvent)
open fun subscription(): Pair<BroadcastChannel<*>, TEvent>? = null
protected fun setState(state: TState) {
if (this.state != state) {
this.state = state
callback?.invoke(state)
}
}
protected fun getState(): TState {
if (state == null)
state = getInitState()
return state!!
}
open fun onStateChanged(callback: ((TState) -> Unit)?) {
this.callback = callback
callback?.invoke(getState())
updateSubscriptions(callback)
}
private fun updateSubscriptions(callback: ((TState) -> Unit)?) {
subscriptionJob?.cancel()
subscriptionJob = null
if (callback != null) {
val (broadcast, updateEvent) = subscription() ?: return
subscriptionJob = launch(UI) {
broadcast.openSubscription()
.use { s ->
while (true) {
s.receive()
update(updateEvent)
}
}
}
}
}
}