Files
letro-ios/ElementX/Sources/Other/UserSettingPropertyWrapper.swift
Stefan Ceriu bf19e3e00b Made room state event collapsing configurable, expose it in the developer menu. Allow the UserSetting property wrappers to only store values in memory (+11 squashed commits)
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
2023-02-24 07:34:07 +02:00

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 }
}