-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRSSApp.swift
88 lines (70 loc) · 2.14 KB
/
RSSApp.swift
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
//
// App.swift
// iosApp
//
// Created by Ekaterina.Petrova on 13.11.2020.
// Copyright © 2020 orgName. All rights reserved.
//
import Foundation
import SwiftUI
import RssReader
@main
class RSSApp: App {
let rss: RssReader
let store: ObservableFeedStore
required init() {
rss = RssReader.Companion().create(withLog: true)
store = ObservableFeedStore(store: FeedStore(rssReader: rss))
}
var body: some Scene {
WindowGroup {
RootView().environmentObject(store)
}
}
}
class ObservableFeedStore: ObservableObject {
@Published public var state: FeedState = FeedState(progress: false, feeds: [], selectedFeed: nil)
@Published public var sideEffect: FeedSideEffect?
let store: FeedStore
var stateWatcher : Closeable?
var sideEffectWatcher : Closeable?
init(store: FeedStore) {
self.store = store
stateWatcher = self.store.watchState().watch { [weak self] state in
self?.state = state
}
sideEffectWatcher = self.store.watchSideEffect().watch { [weak self] state in
self?.sideEffect = state
}
}
public func dispatch(_ action: FeedAction) {
store.dispatch(action: action)
}
deinit {
stateWatcher?.close()
sideEffectWatcher?.close()
}
}
public typealias DispatchFunction = (FeedAction) -> ()
public protocol ConnectedView: View {
associatedtype Props
associatedtype V: View
func map(state: FeedState, dispatch: @escaping DispatchFunction) -> Props
func body(props: Props) -> V
}
public extension ConnectedView {
func render(state: FeedState, dispatch: @escaping DispatchFunction) -> V {
let props = map(state: state, dispatch: dispatch)
return body(props: props)
}
var body: StoreConnector<V> {
return StoreConnector(content: render)
}
}
public struct StoreConnector<V: View>: View {
@EnvironmentObject var store: ObservableFeedStore
let content: (FeedState, @escaping DispatchFunction) -> V
public var body: V {
return content(store.state, store.dispatch)
}
}