-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathQLAnchor.swift
144 lines (115 loc) · 3.95 KB
/
QLAnchor.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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
import Dispatch
public protocol AnyAnchor: AnyObject {
var inputSegment: AnySegment? { get }
}
public final class QLAnchor<Input>: AnyAnchor {
public typealias OnChange = (Input?)->()
public typealias OnError = (Error)->()
public typealias EchoFilter = (Input?, QLAnchor<Input>) -> (Bool)
internal static var DefaultEchoFilter: EchoFilter { return { _, _ in return true } }
public enum Timing {
case early, late
}
internal final class Repeater {
weak var anchor: QLAnchor?
let timing: Timing
init(_ anchor: QLAnchor, timing: Timing) {
self.timing = timing
self.anchor = anchor
}
func echo(value: Input?, filter: EchoFilter, timing: Timing) {
if let repeater = self.anchor,
filter(value, repeater) {
repeater.value = value
}
}
func echo(error: Error) {
anchor?.error = error
}
}
lazy var inputQueue = DispatchQueue(label: "\(self).inputQueue",
qos: .userInitiated)
public required init(onChange: @escaping OnChange,
onError: @escaping OnError) {
self.onChange = onChange
self.onError = onError
}
public var onChange: OnChange
public var onError: OnError
public var inputSegment: AnySegment?
public func addRepeaters(_ repeaters: [QLAnchor], timing: Timing) {
let repeaters = repeaters.map { Repeater($0, timing: timing) }
self._repeaters.append(contentsOf: repeaters)
}
public func getRepeaters(timing: Timing) -> [QLAnchor] {
return _repeaters.compactMap { ($0.timing == timing) ? $0.anchor : nil }
}
internal var _repeaters: [Repeater] = []
public var echoFilter: EchoFilter = DefaultEchoFilter
private var _value: Input?
public var value: Input? {
get {
var safeInput: Input? = nil
inputQueue.sync { safeInput = self._value }
return safeInput
}
set {
inputQueue.sync { self._value = newValue }
if let err = getReroutableError(newValue) {
self.error = err
} else {
echo(value: newValue, timing: .early)
dispatch(value: newValue)
echo(value: newValue, timing: .late)
}
if (QLCommon.Config.Anchor.releaseValues) {
inputQueue.sync { self._value = nil }
}
}
}
private var _error: Error?
public var error: Error? {
get {
var safeError: Error? = nil
inputQueue.sync { safeError = self._error }
return safeError
}
set {
let err: Error = newValue ?? QLCommon.Error.ThrownButNotSet
inputQueue.sync { self._error = err }
dispatch(error: err)
echo(error: err)
if (QLCommon.Config.Anchor.releaseValues) {
inputQueue.sync { self._error = nil }
}
}
}
private func getReroutableError(_ newValue: Input?) -> Error? {
guard QLCommon.Config.Anchor.autoThrowResultFailures,
let errGettable = newValue as? ErrorGettable,
let err = errGettable.getError()
else { return nil }
return err
}
private func dispatch(value: Input?) {
DispatchQueue.main.async {
self.onChange(value)
}
}
private func dispatch(error: Error) {
DispatchQueue.main.async {
self.onError(error)
}
}
private func echo(value: Input?, timing: Timing) {
for repeater in _repeaters {
guard repeater.timing == timing else { continue }
repeater.echo(value: value, filter: echoFilter, timing: timing)
}
}
private func echo(error: Error) {
for repeater in _repeaters {
repeater.echo(error: error)
}
}
}