// // StateWrapper.swift // Meta // // Created by david-swift on 09.06.24. // import Observation /// A storage for `@State` properties. public struct StateWrapper: ConvenienceWidget { /// The content. var content: Body /// The state information (from properties with the `State` wrapper). var state: [String: StateProtocol] = [:] /// The debug tree parameters. public var debugTreeParameters: [(String, value: CustomStringConvertible)] { [ ("state", value: state.map { $0.key.dropFirst() }.joined(separator: ", ")) ] } /// The debug tree's content. public var debugTreeContent: [(String, body: Body)] { [("content", body: content)] } /// The identifier of the field storing whether to update the wrapper's content. private var updateID: String { "update" } /// Initialize a `StateWrapper`. /// - Parameter content: The view content. public init(@ViewBuilder content: @escaping () -> Body) { self.content = content() } /// Initialize a `StateWrapper`. /// - Parameters: /// - content: The view content. /// - state: The state information. init(content: @escaping () -> Body, state: [String: StateProtocol]) { self.content = content() self.state = state } /// Update a view storage. /// - Parameters: /// - storage: The view storage. /// - modifiers: Modify views before being updated. /// - updateProperties: Whether to update properties. /// - type: The type of the widgets. public func update( _ storage: ViewStorage, modifiers: [(AnyView) -> AnyView], updateProperties: Bool, type: WidgetType.Type ) { var updateProperties = storage.fields[updateID] as? Bool ?? false storage.fields[updateID] = false for property in state { if let storage = storage.state[property.key]?.content.storage { property.value.content.storage = storage } if property.value.content.storage.update { updateProperties = true property.value.content.storage.update = false } } guard let storages = storage.content[.mainContent] else { return } content.update(storages, modifiers: modifiers, updateProperties: updateProperties, type: type) } /// Get a view storage. /// - Parameters: /// - modifiers: Modify views before being updated. /// - type: The type of the widgets. /// - Returns: The view storage. public func container(modifiers: [(AnyView) -> AnyView], type: WidgetType.Type) -> ViewStorage { let content = content.storages(modifiers: modifiers, type: type) let storage = ViewStorage(nil, content: [.mainContent: content]) storage.state = state observe(storage: storage) return storage } /// Observe the observable properties accessed in the view. /// - Parameter storage: The view storage func observe(storage: ViewStorage) { withObservationTracking { _ = content.getDebugTree(parameters: true, type: AnyView.self) } onChange: { storage.fields[updateID] = true UpdateManager.updateViews() observe(storage: storage) } } }