forked from square/Listable
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathListStateObserver.swift
257 lines (198 loc) · 8.06 KB
/
ListStateObserver.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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
//
// ListStateObserver.swift
// ListableUI
//
// Created by Kyle Van Essen on 7/9/20.
//
import Foundation
import UIKit
/// Allows reading state and events based on state changes within the list view.
/// For example, you can determine when a user scrolls, when the content of a list
/// changes, etc.
///
/// This is useful if you want to log these events to a logging or debugging system,
/// or potentially perform actions on the list based on some change.
///
/// Every callback has its own data type, filled with information relevant to that callback.
/// Every callback also contains a `ListActions` to perform actions back on the list.
///
/// You can register for each callback type multiple times – eg to split apart different pieces of
/// functionality. Eg, two calls to `onDidScroll` registers two callbacks.
///
/// Example
/// -------
/// ```
/// ListStateObserver { observer in
/// observer.onDidScroll { info in
/// // Called whenever the list is scrolled.
/// }
///
/// observer.onContentChanged { info in
/// // Called when items are inserted or removed.
/// }
/// }
/// ```
/// Note that the duration of performing all callbacks is logged to `os_signpost`. If you find that
/// your application is running slowly, and you have registered `ListStateObserver` callbacks,
/// use Instruments.app to see what callback is slow.
///
public struct ListStateObserver {
/// Creates and optionally allows you to configure an observer.
public init(_ configure : (inout ListStateObserver) -> () = { _ in })
{
configure(&self)
}
//
// MARK: Responding To Scrolling
//
public typealias OnDidScroll = (DidScroll) -> ()
/// Registers a callback which will be called when the list view is scrolled, or is
/// scrolled to top.
///
/// **Note** This callback is called very frequently when the user is scrolling
/// the list. As such, make sure any work you do in the callback is efficient.
public mutating func onDidScroll( _ callback : @escaping OnDidScroll)
{
self.onDidScroll.append(callback)
}
private(set) var onDidScroll : [OnDidScroll] = []
//
// MARK: Responding To Content Updates
//
public typealias OnContentUpdated = (ContentUpdated) -> ()
/// Registers a callback which will be called when the list view's content is updated
/// due to a call to `setContent`.
///
/// **Note**: This method is called even if there were no actual changes made during the `setContent`
/// call. To see if there were changes, check the `hadChanges` property on `ContentUpdated`.
public mutating func onContentUpdated( _ callback : @escaping OnContentUpdated)
{
self.onContentUpdated.append(callback)
}
private(set) var onContentUpdated : [OnContentUpdated] = []
//
// MARK: Responding To Visibility Changes
//
public typealias OnVisibilityChanged = (VisibilityChanged) -> ()
/// Registers a callback which will be called when the list view's content is changed – eg through
/// inserted, removed, updated, moved items or sections.
public mutating func onVisibilityChanged( _ callback : @escaping OnVisibilityChanged)
{
self.onVisibilityChanged.append(callback)
}
private(set) var onVisibilityChanged : [OnVisibilityChanged] = []
//
// MARK: Responding To Frame Changes
//
public typealias OnFrameChanged = (FrameChanged) -> ()
/// Registers a callback which will be called when the list view's frame is changed.
public mutating func onFrameChanged(_ callback : @escaping OnFrameChanged)
{
self.onFrameChanged.append(callback)
}
private(set) var onFrameChanged : [OnFrameChanged] = []
//
// MARK: Responding To Selection Changes
//
public typealias OnSelectionChanged = (SelectionChanged) -> ()
/// Registers a callback which will be called when the list view's selected items are changed by the user.
public mutating func onSelectionChanged(_ callback : @escaping OnSelectionChanged)
{
self.onSelectionChanged.append(callback)
}
private(set) var onSelectionChanged : [OnSelectionChanged] = []
//
// MARK: Responding To Reordered Items
//
public typealias OnItemReordered = (ItemReordered) -> ()
/// Registers a callback which will be called when an item in the list view is reordered by the user.
/// May be called multiple times in a row for reorder events which contain multiple items.
public mutating func onItemReordered(_ callback : @escaping OnItemReordered)
{
self.onItemReordered.append(callback)
}
private(set) var onItemReordered : [OnItemReordered] = []
//
// MARK: Internal Methods
//
static func perform<CallbackInfo>(
_ callbacks : Array<(CallbackInfo) -> ()>,
_ loggingName : StaticString,
with listView : ListView, makeInfo : (ListActions) -> (CallbackInfo)
){
guard callbacks.isEmpty == false else {
return
}
let actions = ListActions()
actions.listView = listView
let callbackInfo = makeInfo(actions)
SignpostLogger.log(log: .stateObserver, name: loggingName, for: listView) {
callbacks.forEach {
$0(callbackInfo)
}
}
actions.listView = nil
}
}
extension ListStateObserver
{
/// Parameters available for ``OnDidScroll`` callbacks.
public struct DidScroll {
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
}
/// Parameters available for ``OnContentUpdated`` callbacks.
public struct ContentUpdated {
public let hadChanges : Bool
public let insertionsAndRemovals : InsertionsAndRemovals
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
public struct InsertionsAndRemovals {
public var sections : ChangedIDs
public var items : ChangedIDs
init(diff : SectionedDiff<Section, AnyIdentifier, AnyItem, AnyIdentifier>) {
self.sections = ChangedIDs(
inserted: Set(diff.changes.added.map{ $0.identifier }),
removed: Set(diff.changes.removed.map{ $0.identifier })
)
self.items = ChangedIDs(
inserted: diff.changes.addedItemIdentifiers,
removed: diff.changes.removedItemIdentifiers
)
}
public struct ChangedIDs {
public var inserted : Set<AnyIdentifier>
public var removed : Set<AnyIdentifier>
}
}
}
/// Parameters available for ``OnVisibilityChanged`` callbacks.
public struct VisibilityChanged {
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
public let displayed : [AnyItem]
public let endedDisplay : [AnyItem]
}
/// Parameters available for ``OnFrameChanged`` callbacks.
public struct FrameChanged {
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
public let old : CGRect
public let new : CGRect
}
/// Parameters available for ``OnSelectionChanged`` callbacks.
public struct SelectionChanged {
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
public let old : Set<AnyIdentifier>
public let new : Set<AnyIdentifier>
}
/// Parameters available for ``OnItemReordered`` callbacks.
public struct ItemReordered {
public let actions : ListActions
public let positionInfo : ListScrollPositionInfo
public let item : AnyItem
public let sections : [Section]
public let result : ItemReordering.Result
}
}