Skip to content
Snippets Groups Projects
Select Git revision
  • 4838dfdd91f142f06aa5d2d2402505fd61df672d
  • main default protected
  • development
  • integration
  • v1.1.5
  • v1.1.4
  • v1.1.3
  • v1.1.2
  • v1.1.1
  • v1.1.0
  • v1.0.0
11 results

ConnectionCloser.swift

Blame
  • AppView.swift 3.58 KiB
    import ComposableArchitecture
    import HomeFeature
    import RestoreFeature
    import SwiftUI
    import WelcomeFeature
    
    struct AppView: View {
      let store: Store<AppState, AppAction>
    
      enum ViewState: Equatable {
        case loading
        case welcome
        case restore
        case home
        case failure(String)
    
        init(_ state: AppState) {
          switch state.screen {
          case .loading: self = .loading
          case .welcome(_): self = .welcome
          case .restore(_): self = .restore
          case .home(_): self = .home
          case .failure(let failure): self = .failure(failure)
          }
        }
      }
    
      var body: some View {
        WithViewStore(store.scope(state: ViewState.init)) { viewStore in
          ZStack {
            switch viewStore.state {
            case .loading:
              ProgressView {
                Text("Loading")
              }
              .controlSize(.large)
              .frame(maxWidth: .infinity, maxHeight: .infinity)
              .transition(.opacity)
    
            case .welcome:
              IfLetStore(
                store.scope(
                  state: { (/AppState.Screen.welcome).extract(from: $0.screen) },
                  action: AppAction.welcome
                ),
                then: { store in
                  WelcomeView(store: store)
                    .frame(maxWidth: .infinity, maxHeight: .infinity)
                    .transition(.asymmetric(
                      insertion: .move(edge: .trailing),
                      removal: .opacity
                    ))
                }
              )
    
            case .restore:
              IfLetStore(
                store.scope(
                  state: { (/AppState.Screen.restore).extract(from: $0.screen) },
                  action: AppAction.restore
                ),
                then: { store in
                  RestoreView(store: store)
                    .frame(maxWidth: .infinity, maxHeight: .infinity)
                    .transition(.asymmetric(
                      insertion: .move(edge: .trailing),
                      removal: .opacity
                    ))
                }
              )
    
            case .home:
              IfLetStore(
                store.scope(
                  state: { (/AppState.Screen.home).extract(from: $0.screen) },
                  action: AppAction.home
                ),
                then: { store in
                  HomeView(store: store)
                    .frame(maxWidth: .infinity, maxHeight: .infinity)
                    .transition(.asymmetric(
                      insertion: .move(edge: .trailing),
                      removal: .opacity
                    ))
                }
              )
    
            case .failure(let failure):
              NavigationView {
                VStack(spacing: 0) {
                  ScrollView {
                    Text(failure)
                      .frame(maxWidth: .infinity, alignment: .leading)
                      .padding()
                  }
    
                  Divider()
    
                  Button {
                    viewStore.send(.start)
                  } label: {
                    Text("Retry")
                      .frame(maxWidth: .infinity)
                  }
                  .buttonStyle(.borderedProminent)
                  .controlSize(.large)
                  .padding()
                }
                .navigationTitle("Error")
              }
              .navigationViewStyle(.stack)
              .frame(maxWidth: .infinity, maxHeight: .infinity)
              .transition(.asymmetric(
                insertion: .move(edge: .trailing),
                removal: .opacity
              ))
            }
          }
          .animation(.default, value: viewStore.state)
          .task { viewStore.send(.start) }
        }
      }
    }
    
    #if DEBUG
    struct AppView_Previews: PreviewProvider {
      static var previews: some View {
        AppView(store: Store(
          initialState: AppState(),
          reducer: .empty,
          environment: ()
        ))
      }
    }
    #endif