-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathLoginFeature.swift
More file actions
123 lines (107 loc) · 3.4 KB
/
Copy pathLoginFeature.swift
File metadata and controls
123 lines (107 loc) · 3.4 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
//
// LoginFeature.swift
// DevLogPresentation
//
// Created by opfic on 6/5/26.
//
import ComposableArchitecture
import DevLogDomain
import Foundation
@Reducer
struct LoginFeature {
@ObservableState
struct State: Equatable {
@Presents var alert: AlertState<Never>?
var loading = LoadingFeature.State()
var isLoading: Bool {
loading.isLoading
}
}
enum Action {
case alert(PresentationAction<Never>)
case tapSignInButton(AuthProvider)
case signInFailed(AlertType)
case loading(LoadingFeature.Action)
}
enum AlertType: Equatable {
case emailUnavailable
case error
}
@Dependency(\.signInUseCase) var signInUseCase
var body: some ReducerOf<Self> {
Scope(state: \.loading, action: \.loading) {
LoadingFeature()
}
Reduce { state, action in
switch action {
case .alert:
break
case .tapSignInButton(let provider):
return signInEffect(provider)
case .signInFailed(let alertType):
state.alert = Self.alertState(for: alertType)
case .loading:
break
}
return .none
}
.ifLet(\.$alert, action: \.alert)
}
}
extension DependencyValues {
var signInUseCase: SignInUseCase {
get { self[SignInUseCaseKey.self] }
set { self[SignInUseCaseKey.self] = newValue }
}
}
private enum SignInUseCaseKey: DependencyKey {
static var liveValue: SignInUseCase {
preconditionFailure("SignInUseCase must be provided.")
}
static var testValue: SignInUseCase {
liveValue
}
}
private extension LoginFeature {
func signInEffect(_ provider: AuthProvider) -> Effect<Action> {
.run { [signInUseCase] send in
await send(.loading(.begin(target: .default, mode: .immediate)))
do {
let signedIn = try await signInUseCase.execute(provider)
// 유스케이스 완료가 화면 전환 완료를 의미하지 않으므로 LoginView가 교체될 때까지 로딩을 유지한다.
guard !signedIn else { return }
await send(.loading(.end(target: .default, mode: .immediate)))
} catch {
await send(.loading(.end(target: .default, mode: .immediate)))
await send(.signInFailed(Self.alertType(for: error)))
}
}
}
static func alertState(for alertType: AlertType) -> AlertState<Never> {
let title: String
let message: String
switch alertType {
case .emailUnavailable:
title = String(localized: "login_alert_email_unavailable_title")
message = String(localized: "login_alert_email_unavailable_message")
case .error:
title = String(localized: "common_error_title")
message = String(localized: "common_error_message")
}
return AlertState {
TextState(title)
} actions: {
ButtonState(role: .cancel) {
TextState(String(localized: "common_close"))
}
} message: {
TextState(message)
}
}
static func alertType(for error: Error) -> AlertType {
if case AuthError.emailNotFound = error {
return .emailUnavailable
}
return .error
}
}