Tusker/Packages/TuskerPreferences/Sources/TuskerPreferences/Preference.swift

93 lines
2.7 KiB
Swift
Raw Normal View History

2024-04-13 22:44:43 +00:00
//
// Preference.swift
// TuskerPreferences
//
// Created by Shadowfacts on 4/13/24.
//
import Foundation
import Combine
// TODO: once we target iOS 17, use Observable for this
@propertyWrapper
final class Preference<Key: PreferenceKey>: Codable {
@Published private(set) var storedValue: Key.Value?
var wrappedValue: Key.Value {
get {
storedValue ?? Key.defaultValue
}
set {
fatalError("unreachable")
}
}
init() {
self.storedValue = nil
}
init(from decoder: any Decoder) throws {
if let container = try? decoder.singleValueContainer() {
self.storedValue = try? container.decode(Key.Value.self)
}
}
func encode(to encoder: any Encoder) throws {
if let storedValue {
var container = encoder.singleValueContainer()
try container.encode(storedValue)
}
}
static subscript(
_enclosingInstance instance: PreferenceStore,
wrapped wrappedKeyPath: ReferenceWritableKeyPath<PreferenceStore, Key.Value>,
storage storageKeyPath: ReferenceWritableKeyPath<PreferenceStore, Preference>
) -> Key.Value {
get {
2024-04-14 02:36:42 +00:00
get(enclosingInstance: instance, storage: storageKeyPath)
2024-04-13 22:44:43 +00:00
}
set {
2024-04-14 02:36:42 +00:00
set(enclosingInstance: instance, storage: storageKeyPath, newValue: newValue)
2024-04-13 22:44:43 +00:00
Key.didSet(in: instance, newValue: newValue)
}
}
// for testing only
@inline(__always)
static func get<Enclosing>(
enclosingInstance: Enclosing,
2024-04-14 02:36:42 +00:00
storage: KeyPath<Enclosing, Preference>
2024-04-13 22:44:43 +00:00
) -> Key.Value where Enclosing: ObservableObject, Enclosing.ObjectWillChangePublisher == ObservableObjectPublisher {
let pref = enclosingInstance[keyPath: storage]
return pref.storedValue ?? Key.defaultValue
}
// for testing only
@inline(__always)
static func set<Enclosing>(
enclosingInstance: Enclosing,
2024-04-14 02:36:42 +00:00
storage: KeyPath<Enclosing, Preference>,
2024-04-13 22:44:43 +00:00
newValue: Key.Value
) where Enclosing: ObservableObject, Enclosing.ObjectWillChangePublisher == ObservableObjectPublisher {
enclosingInstance.objectWillChange.send()
let pref = enclosingInstance[keyPath: storage]
pref.storedValue = newValue
}
2024-04-14 02:36:42 +00:00
var projectedValue: PreferencePublisher<Key> {
.init(preference: self)
}
}
2024-04-13 22:44:43 +00:00
2024-04-14 02:36:42 +00:00
struct PreferencePublisher<Key: PreferenceKey>: Publisher {
typealias Output = Key.Value
typealias Failure = Never
let preference: Preference<Key>
func receive<S>(subscriber: S) where S : Subscriber, Never == S.Failure, Key.Value == S.Input {
preference.$storedValue.map { $0 ?? Key.defaultValue }.receive(subscriber: subscriber)
2024-04-13 22:44:43 +00:00
}
}