Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/babylonhealth/reactivefeedback
Unidirectional reactive architecture
https://github.com/babylonhealth/reactivefeedback
architecture ios reactiveswift redux swift unidirectional-data-flow
Last synced: about 1 month ago
JSON representation
Unidirectional reactive architecture
- Host: GitHub
- URL: https://github.com/babylonhealth/reactivefeedback
- Owner: babylonhealth
- License: mit
- Created: 2017-08-29T23:09:17.000Z (over 7 years ago)
- Default Branch: develop
- Last Pushed: 2023-05-04T13:19:20.000Z (over 1 year ago)
- Last Synced: 2024-12-10T05:32:52.354Z (about 1 month ago)
- Topics: architecture, ios, reactiveswift, redux, swift, unidirectional-data-flow
- Language: Swift
- Homepage:
- Size: 229 KB
- Stars: 158
- Watchers: 50
- Forks: 11
- Open Issues: 4
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# ReactiveFeedback
Unidirectional Reactive Architecture. This is a [ReactiveSwift](https://github.com/ReactiveCocoa/ReactiveSwift) implemetation of [RxFeedback](https://github.com/kzaher/RxFeedback)
## Documentation
![](diagrams/ReactiveFeedback.jpg)
### Motivation
Requirements for iOS apps have become huge. Our code has to manage a lot of state e.g. server responses, cached data, UI state, routing etc. Some may say that Reactive Programming can help us a lot but, in the wrong hands, it can do even more harm to your code base.
The goal of this library is to provide a simple and intuitive approach to designing reactive state machines.
### Core Concepts
##### State
`State` is the single source of truth. It represents a state of your system and is usually a plain Swift type (which doesn't contain any ReactiveSwift primitives). Your state is immutable. The only way to transition from one `State` to another is to emit an `Event`.
```swift
struct Results {
let page: Int
let totalResults: Int
let totalPages: Int
let results: [T]static func empty() -> Results {
return Results(page: 0, totalResults: 0, totalPages: 0, results: [])
}
}struct Context {
var batch: Results
var movies: [Movie]static var empty: Context {
return Context(batch: Results.empty(), movies: [])
}
}enum State {
case initial
case paging(context: Context)
case loadedPage(context: Context)
case refreshing(context: Context)
case refreshed(context: Context)
case error(error: NSError, context: Context)
case retry(context: Context)
}
```##### Event
Represents all possible events that can happen in your system which can cause a transition to a new `State`.
```swift
enum Event {
case startLoadingNextPage
case response(Results)
case failed(NSError)
case retry
}
```##### Reducer
A Reducer is a pure function with a signature of `(State, Event) -> State`. While `Event` represents an action that results in a `State` change, it's actually not what _causes_ the change. An `Event` is just that, a representation of the intention to transition from one state to another. What actually causes the `State` to change, the embodiment of the corresponding `Event`, is a Reducer. A Reducer is the only place where a `State` can be changed.
```swift
static func reduce(state: State, event: Event) -> State {
switch event {
case .startLoadingNextPage:
return .paging(context: state.context)
case .response(let batch):
var copy = state.context
copy.batch = batch
copy.movies += batch.results
return .loadedPage(context: copy)
case .failed(let error):
return .error(error: error, context: state.context)
case .retry:
return .retry(context: state.context)
}
}
```##### Feedback
While `State` represents where the system is at a given time, `Event` represents a trigger for state change, and a `Reducer` is the pure function that changes the state depending on current state and type of event received, there is not as of yet any type to emit events given a particular current state. That's the job of the `Feedback`. It's essentially a "processing engine", listening to changes in the current `State` and emitting the corresponding next events to take place. It's represented by a pure function with a signature of `Signal -> Signal`. Feedbacks don't directly mutate states. Instead, they only emit events which then cause states to change in reducers.
```swift
public struct Feedback {
public let events: (Scheduler, Signal) -> Signal
}func loadNextFeedback(for nearBottomSignal: Signal) -> Feedback {
return Feedback(predicate: { !$0.paging }) { _ in
return nearBottomSignal
.map { Event.startLoadingNextPage }
}
}func pagingFeedback() -> Feedback {
return Feedback(skippingRepeated: { $0.nextPage }) { (nextPage) -> SignalProducer in
return URLSession.shared.fetchMovies(page: nextPage)
.map(Event.response)
.flatMapError { (error) -> SignalProducer in
return SignalProducer(value: Event.failed(error))
}
}
}func retryFeedback(for retrySignal: Signal) -> Feedback {
return Feedback(skippingRepeated: { $0.lastError }) { _ -> Signal in
return retrySignal.map { Event.retry }
}
}func retryPagingFeedback() -> Feedback {
return Feedback(skippingRepeated: { $0.retryPage }) { (nextPage) -> SignalProducer in
return URLSession.shared.fetchMovies(page: nextPage)
.map(Event.response)
.flatMapError { (error) -> SignalProducer in
return SignalProducer(value: Event.failed(error))
}
}
}
```### The Flow
1. As you can see from the diagram above we always start with an initial state.
2. Every change to the `State` will be then delivered to all `Feedback` loops that were added to the system.
3. `Feedback` then decides whether any action should be performed with a subset of the `State` (e.g calling API, observe UI events) by dispatching an `Event`, or ignoring it by returning `SignalProducer.empty`.
4. Dispatched `Event` then goes to the `Reducer` which applies it and returns a new value of the `State`.
5. And then cycle starts all over (see 2).##### Example
```swift
let increment = Feedback { _ in
return self.plusButton.reactive
.controlEvents(.touchUpInside)
.map { _ in Event.increment }
}let decrement = Feedback { _ in
return self.minusButton.reactive
.controlEvents(.touchUpInside)
.map { _ in Event.decrement }
}let system = SignalProducer.system(initial: 0,
reduce: { (count, event) -> Int in
switch event {
case .increment:
return count + 1
case .decrement:
return count - 1
}
},
feedbacks: [increment, decrement])label.reactive.text <~ system.map(String.init)
```![](diagrams/increment_example.gif)
### Advantages
TODO