mirror of
https://github.com/pointfreeco/swift-composable-architecture.git
synced 2025-12-24 12:14:25 +01:00
* Add an `EffectOf<Action>` typealias for `Effect<Action, Never>` * Fix doc * Rename `EffectOf` to `EffectTask` * Rename `Effect` to `EffectPublisher` * Soft-deprecate `Effect` * Link to `EffectTask` * Use `EffectPublisher` in Combine contexts * Reword soft-deprecation message * Remove `renamed:` fix-it for `Effect` deprecation * Update Sources/ComposableArchitecture/Documentation.docc/Extensions/ReducerProtocol.md Co-authored-by: Stephen Celis <stephen.celis@gmail.com> * Update Sources/ComposableArchitecture/Documentation.docc/ComposableArchitecture.md * Update Sources/ComposableArchitecture/Effect.swift * Fix DocC identifiers Co-authored-by: Stephen Celis <stephen.celis@gmail.com>
96 lines
2.0 KiB
Swift
96 lines
2.0 KiB
Swift
import ComposableArchitecture
|
|
import SwiftUI
|
|
|
|
private let readMe = """
|
|
This screen demonstrates the basics of the Composable Architecture in an archetypal counter \
|
|
application.
|
|
|
|
The domain of the application is modeled using simple data types that correspond to the mutable \
|
|
state of the application and any actions that can affect that state or the outside world.
|
|
"""
|
|
|
|
// MARK: - Feature domain
|
|
|
|
struct Counter: ReducerProtocol {
|
|
struct State: Equatable {
|
|
var count = 0
|
|
}
|
|
|
|
enum Action: Equatable {
|
|
case decrementButtonTapped
|
|
case incrementButtonTapped
|
|
}
|
|
|
|
func reduce(into state: inout State, action: Action) -> EffectTask<Action> {
|
|
switch action {
|
|
case .decrementButtonTapped:
|
|
state.count -= 1
|
|
return .none
|
|
case .incrementButtonTapped:
|
|
state.count += 1
|
|
return .none
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Feature view
|
|
|
|
struct CounterView: View {
|
|
let store: StoreOf<Counter>
|
|
|
|
var body: some View {
|
|
WithViewStore(self.store, observe: { $0 }) { viewStore in
|
|
HStack {
|
|
Button {
|
|
viewStore.send(.decrementButtonTapped)
|
|
} label: {
|
|
Image(systemName: "minus")
|
|
}
|
|
|
|
Text("\(viewStore.count)")
|
|
.monospacedDigit()
|
|
|
|
Button {
|
|
viewStore.send(.incrementButtonTapped)
|
|
} label: {
|
|
Image(systemName: "plus")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
struct CounterDemoView: View {
|
|
let store: StoreOf<Counter>
|
|
|
|
var body: some View {
|
|
Form {
|
|
Section {
|
|
AboutView(readMe: readMe)
|
|
}
|
|
|
|
Section {
|
|
CounterView(store: self.store)
|
|
.frame(maxWidth: .infinity)
|
|
}
|
|
}
|
|
.buttonStyle(.borderless)
|
|
.navigationTitle("Counter demo")
|
|
}
|
|
}
|
|
|
|
// MARK: - SwiftUI previews
|
|
|
|
struct CounterView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
NavigationView {
|
|
CounterDemoView(
|
|
store: Store(
|
|
initialState: Counter.State(),
|
|
reducer: Counter()
|
|
)
|
|
)
|
|
}
|
|
}
|
|
}
|