mirror of
https://github.com/pointfreeco/swift-composable-architecture.git
synced 2025-12-20 09:11:33 +01:00
* wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Silence test warnings * wip * wip * wip * update a bunch of docs * wip * wip * fix * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Kill integration tests for now * wip * wip * wip * wip * updating docs for @Reducer macro * replaced more Reducer protocols with @Reducer * Fixed some broken docc references * wip * Some @Reducer docs * more docs * convert some old styles to new style * wip * wip * wip * wip * wip * wip * wip * bump * update tutorials to use body * update tutorials to use DML on destination state enum * Add diagnostic * wip * updated a few more tests * wip * wip * Add another gotcha * wip * wip * wip * fixes * wip * wip * wip * wip * wip * fix * wip * remove for now * wip * wip * updated some docs * migration guides * more migration guide * fix ci * fix * soft deprecate all apis using AnyCasePath * wip * Fix * fix tests * swift-format 509 compatibility * wip * wip * Update Sources/ComposableArchitecture/Macros.swift Co-authored-by: Mateusz Bąk <bakmatthew@icloud.com> * wip * wip * update optional state case study * remove initializer * Don't use @State for BasicsView integration demo * fix tests * remove reduce diagnostics for now * diagnose error not warning * Update Sources/ComposableArchitecture/Macros.swift Co-authored-by: Jesse Tipton <jesse@jessetipton.com> * wip * move integration tests to cron * Revert "move integration tests to cron" This reverts commitf9bdf2f04b. * disable flakey tests on CI * wip * wip * Revert "Revert "move integration tests to cron"" This reverts commit66aafa7327. * fix * wip * fix --------- Co-authored-by: Brandon Williams <mbrandonw@hey.com> Co-authored-by: Mateusz Bąk <bakmatthew@icloud.com> Co-authored-by: Brandon Williams <135203+mbrandonw@users.noreply.github.com> Co-authored-by: Jesse Tipton <jesse@jessetipton.com>
148 lines
4.4 KiB
Swift
148 lines
4.4 KiB
Swift
import ComposableArchitecture
|
|
import SwiftUI
|
|
|
|
private let readMe = """
|
|
This demonstrates how to best handle alerts and confirmation dialogs in the Composable \
|
|
Architecture.
|
|
|
|
Because the library demands that all data flow through the application in a single direction, we \
|
|
cannot leverage SwiftUI's two-way bindings because they can make changes to state without going \
|
|
through a reducer. This means we can't directly use the standard API to display alerts and sheets.
|
|
|
|
However, the library comes with two types, `AlertState` and `ConfirmationDialogState`, which can \
|
|
be constructed from reducers and control whether or not an alert or confirmation dialog is \
|
|
displayed. Further, it automatically handles sending actions when you tap their buttons, which \
|
|
allows you to properly handle their functionality in the reducer rather than in two-way bindings \
|
|
and action closures.
|
|
|
|
The benefit of doing this is that you can get full test coverage on how a user interacts with \
|
|
alerts and dialogs in your application
|
|
"""
|
|
|
|
// MARK: - Feature domain
|
|
|
|
@Reducer
|
|
struct AlertAndConfirmationDialog {
|
|
struct State: Equatable {
|
|
@PresentationState var alert: AlertState<Action.Alert>?
|
|
@PresentationState var confirmationDialog: ConfirmationDialogState<Action.ConfirmationDialog>?
|
|
var count = 0
|
|
}
|
|
|
|
enum Action {
|
|
case alert(PresentationAction<Alert>)
|
|
case alertButtonTapped
|
|
case confirmationDialog(PresentationAction<ConfirmationDialog>)
|
|
case confirmationDialogButtonTapped
|
|
|
|
enum Alert {
|
|
case incrementButtonTapped
|
|
}
|
|
enum ConfirmationDialog {
|
|
case incrementButtonTapped
|
|
case decrementButtonTapped
|
|
}
|
|
}
|
|
|
|
var body: some Reducer<State, Action> {
|
|
Reduce { state, action in
|
|
switch action {
|
|
case .alert(.presented(.incrementButtonTapped)),
|
|
.confirmationDialog(.presented(.incrementButtonTapped)):
|
|
state.alert = AlertState { TextState("Incremented!") }
|
|
state.count += 1
|
|
return .none
|
|
|
|
case .alert:
|
|
return .none
|
|
|
|
case .alertButtonTapped:
|
|
state.alert = AlertState {
|
|
TextState("Alert!")
|
|
} actions: {
|
|
ButtonState(role: .cancel) {
|
|
TextState("Cancel")
|
|
}
|
|
ButtonState(action: .incrementButtonTapped) {
|
|
TextState("Increment")
|
|
}
|
|
} message: {
|
|
TextState("This is an alert")
|
|
}
|
|
return .none
|
|
|
|
case .confirmationDialog(.presented(.decrementButtonTapped)):
|
|
state.alert = AlertState { TextState("Decremented!") }
|
|
state.count -= 1
|
|
return .none
|
|
|
|
case .confirmationDialog:
|
|
return .none
|
|
|
|
case .confirmationDialogButtonTapped:
|
|
state.confirmationDialog = ConfirmationDialogState {
|
|
TextState("Confirmation dialog")
|
|
} actions: {
|
|
ButtonState(role: .cancel) {
|
|
TextState("Cancel")
|
|
}
|
|
ButtonState(action: .incrementButtonTapped) {
|
|
TextState("Increment")
|
|
}
|
|
ButtonState(action: .decrementButtonTapped) {
|
|
TextState("Decrement")
|
|
}
|
|
} message: {
|
|
TextState("This is a confirmation dialog.")
|
|
}
|
|
return .none
|
|
}
|
|
}
|
|
.ifLet(\.$alert, action: \.alert)
|
|
.ifLet(\.$confirmationDialog, action: \.confirmationDialog)
|
|
}
|
|
}
|
|
|
|
// MARK: - Feature view
|
|
|
|
struct AlertAndConfirmationDialogView: View {
|
|
@State var store = Store(initialState: AlertAndConfirmationDialog.State()) {
|
|
AlertAndConfirmationDialog()
|
|
}
|
|
|
|
var body: some View {
|
|
WithViewStore(self.store, observe: { $0 }) { viewStore in
|
|
Form {
|
|
Section {
|
|
AboutView(readMe: readMe)
|
|
}
|
|
|
|
Text("Count: \(viewStore.count)")
|
|
Button("Alert") { viewStore.send(.alertButtonTapped) }
|
|
Button("Confirmation Dialog") { viewStore.send(.confirmationDialogButtonTapped) }
|
|
}
|
|
}
|
|
.navigationTitle("Alerts & Dialogs")
|
|
.alert(
|
|
store: self.store.scope(state: \.$alert, action: { .alert($0) })
|
|
)
|
|
.confirmationDialog(
|
|
store: self.store.scope(state: \.$confirmationDialog, action: { .confirmationDialog($0) })
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - SwiftUI previews
|
|
|
|
struct AlertAndConfirmationDialog_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
NavigationView {
|
|
AlertAndConfirmationDialogView(
|
|
store: Store(initialState: AlertAndConfirmationDialog.State()) {
|
|
AlertAndConfirmationDialog()
|
|
}
|
|
)
|
|
}
|
|
}
|
|
}
|