Squashed commits: [42e45fc] Even more tweaks following code review [5dcd5be] Add swift-algoritms and switch bubbling detection to its `chunked` method [4ac70ed] Move the groupBy implementation to Collection instead of Array [6aeffc3] Tweaks following code review [0ca5ac2] Bubbles working again, grouping computed closer to the UI level [3a66030] Refactor how timeline items are built in the RoomTimelineController [57d51e9] Remove grouping from timeline items [8608950] Remove the RoomTimelineViewFactory, update the GroupRoomTimelineView [9e52e61] Add array grouping extension, unit tests and switched the timeline controller to it [7d213a1] First attempt [90bb1d7] Remove now unused `RoomTimelineController` `updatedTimelineItem` calback
140 lines
4.7 KiB
Swift
140 lines
4.7 KiB
Swift
//
|
|
// Copyright 2022 New Vector Ltd
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
// Taken from https://www.swiftbysundell.com/articles/property-wrappers-in-swift/
|
|
|
|
import Combine
|
|
import Foundation
|
|
|
|
/// Property wrapper that allows transparent access to user defaults and exposes
|
|
/// a combine publisher for listening to value changes
|
|
///
|
|
/// Please use `UserSettingRawRepresentable` for storing RawRepresentable values
|
|
@propertyWrapper
|
|
struct UserSetting<Value: Equatable> {
|
|
private let key: String
|
|
private var defaultValue: Value
|
|
private let storage: UserDefaults?
|
|
private let publisher: CurrentValueSubject<Value, Never>
|
|
|
|
init(key: String, defaultValue: Value, persistIn storage: UserDefaults?) {
|
|
self.key = key
|
|
self.defaultValue = defaultValue
|
|
self.storage = storage
|
|
|
|
let value = storage?.value(forKey: key) as? Value ?? defaultValue
|
|
publisher = CurrentValueSubject<Value, Never>(value)
|
|
}
|
|
|
|
var wrappedValue: Value {
|
|
get {
|
|
let value = storage?.value(forKey: key) as? Value
|
|
return value ?? defaultValue
|
|
}
|
|
set {
|
|
guard let storage else {
|
|
defaultValue = newValue
|
|
publisher.send(defaultValue)
|
|
return
|
|
}
|
|
|
|
if let optional = newValue as? AnyOptional, optional.isNil {
|
|
storage.removeObject(forKey: key)
|
|
publisher.send(defaultValue)
|
|
} else {
|
|
storage.setValue(newValue, forKey: key)
|
|
publisher.send(newValue)
|
|
}
|
|
}
|
|
}
|
|
|
|
var projectedValue: AnyPublisher<Value, Never> {
|
|
publisher.removeDuplicates(by: { $0 == $1 }).eraseToAnyPublisher()
|
|
}
|
|
}
|
|
|
|
extension UserSetting where Value: ExpressibleByNilLiteral {
|
|
init(key: String, persistIn storage: UserDefaults?) {
|
|
self.init(key: key, defaultValue: nil, persistIn: storage)
|
|
}
|
|
}
|
|
|
|
/// Property wrapper that allows transparent access to user defaults for RawRepresentable types
|
|
/// and exposes a combine publisher for listening to value changes
|
|
///
|
|
/// Tried extending UserSetting with RawRepresentable conformance but in that case the non-restricted
|
|
/// method takes precedence. Decided to go with with the simple solution instead of fighting the system
|
|
@propertyWrapper
|
|
struct UserSettingRawRepresentable<Value: RawRepresentable & Equatable> {
|
|
private let key: String
|
|
private var defaultValue: Value
|
|
private let storage: UserDefaults?
|
|
private let publisher: CurrentValueSubject<Value, Never>
|
|
|
|
init(key: String, defaultValue: Value, persistIn storage: UserDefaults?) {
|
|
self.key = key
|
|
self.defaultValue = defaultValue
|
|
self.storage = storage
|
|
|
|
let value = (storage?.value(forKey: key) as? Value.RawValue).flatMap { Value(rawValue: $0) } ?? defaultValue
|
|
publisher = CurrentValueSubject<Value, Never>(value)
|
|
}
|
|
|
|
var wrappedValue: Value {
|
|
get {
|
|
guard let value = storage?.value(forKey: key) as? Value.RawValue else {
|
|
return defaultValue
|
|
}
|
|
|
|
return Value(rawValue: value) ?? defaultValue
|
|
}
|
|
set {
|
|
guard let storage else {
|
|
defaultValue = newValue
|
|
publisher.send(defaultValue)
|
|
return
|
|
}
|
|
|
|
if let optional = newValue as? AnyOptional, optional.isNil {
|
|
storage.removeObject(forKey: key)
|
|
publisher.send(newValue)
|
|
} else {
|
|
storage.setValue(newValue.rawValue, forKey: key)
|
|
publisher.send(newValue)
|
|
}
|
|
}
|
|
}
|
|
|
|
var projectedValue: AnyPublisher<Value, Never> {
|
|
publisher.removeDuplicates(by: { $0 == $1 }).eraseToAnyPublisher()
|
|
}
|
|
}
|
|
|
|
extension UserSettingRawRepresentable where Value: ExpressibleByNilLiteral {
|
|
init(key: String, persistIn storage: UserDefaults?) {
|
|
self.init(key: key, defaultValue: nil, persistIn: storage)
|
|
}
|
|
}
|
|
|
|
// Casting to AnyOptional will fail for any types that are not Optional (below)
|
|
private protocol AnyOptional {
|
|
var isNil: Bool { get }
|
|
}
|
|
|
|
extension Optional: AnyOptional {
|
|
var isNil: Bool { self == nil }
|
|
}
|