Files
swift-composable-architectu…/Examples/CaseStudies/SwiftUICaseStudies/03-Navigation-NavigateAndLoad.swift
2020-06-08 17:40:46 -04:00

107 lines
2.9 KiB
Swift

import Combine
import ComposableArchitecture
import SwiftUI
import UIKit
private let readMe = """
This screen demonstrates navigation that depends on loading optional state.
Tapping "Load optional counter" simultaneously navigates to a screen that depends on optional \
counter state and fires off an effect that will load this state a second later.
"""
struct NavigateAndLoadState: Equatable {
var isNavigationActive = false
var optionalCounter: CounterState?
}
enum NavigateAndLoadAction: Equatable {
case optionalCounter(CounterAction)
case setNavigation(isActive: Bool)
case setNavigationIsActiveDelayCompleted
}
struct NavigateAndLoadEnvironment {
var mainQueue: AnySchedulerOf<DispatchQueue>
}
let navigateAndLoadReducer = counterReducer
.optional
.pullback(
state: \.optionalCounter,
action: /NavigateAndLoadAction.optionalCounter,
environment: { _ in CounterEnvironment() }
)
.combined(
with: Reducer<
NavigateAndLoadState, NavigateAndLoadAction, NavigateAndLoadEnvironment
> { state, action, environment in
switch action {
case .setNavigation(isActive: true):
state.isNavigationActive = true
return Effect(value: .setNavigationIsActiveDelayCompleted)
.delay(for: 1, scheduler: environment.mainQueue)
.eraseToEffect()
case .setNavigation(isActive: false):
state.isNavigationActive = false
state.optionalCounter = nil
return .none
case .setNavigationIsActiveDelayCompleted:
state.optionalCounter = CounterState()
return .none
case .optionalCounter:
return .none
}
}
)
struct NavigateAndLoadView: View {
let store: Store<NavigateAndLoadState, NavigateAndLoadAction>
var body: some View {
WithViewStore(self.store) { viewStore in
Form {
Section(header: Text(readMe)) {
NavigationLink(
destination: IfLetStore(
self.store.scope(
state: { $0.optionalCounter }, action: NavigateAndLoadAction.optionalCounter),
then: CounterView.init(store:),
else: ActivityIndicator()
),
isActive: viewStore.binding(
get: { $0.isNavigationActive },
send: NavigateAndLoadAction.setNavigation(isActive:)
)
) {
HStack {
Text("Load optional counter")
}
}
}
}
}
.navigationBarTitle("Navigate and load")
}
}
struct NavigateAndLoadView_Previews: PreviewProvider {
static var previews: some View {
NavigationView {
NavigateAndLoadView(
store: Store(
initialState: NavigateAndLoadState(),
reducer: navigateAndLoadReducer,
environment: NavigateAndLoadEnvironment(
mainQueue: DispatchQueue.main.eraseToAnyScheduler()
)
)
)
}
.navigationViewStyle(StackNavigationViewStyle())
}
}