-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathLoginViewModel.swift
More file actions
123 lines (110 loc) · 3.36 KB
/
LoginViewModel.swift
File metadata and controls
123 lines (110 loc) · 3.36 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
//
// LoginViewModel.swift
// DevLog
//
// Created by 최윤진 on 11/14/25.
//
import Combine
import Foundation
import FirebaseAuth
import GoogleSignIn
@Observable
final class LoginViewModel: Store {
struct State: Equatable {
var signIn: Bool?
var isLoading = false
var showAlert: Bool = false
var alertTitle: String = ""
var alertMessage: String = ""
}
enum Action {
case signOutAuto
case setAlert(Bool)
case tapSignInButton(AuthProvider)
case tapSignOutButton
case setLoading(Bool)
case setLogined(Bool)
}
enum SideEffect {
case signIn(AuthProvider)
case signOut
}
private let signInUseCase: SignInUseCase
private let signOutUseCase: SignOutUseCase
private let sessionUseCase: AuthSessionUseCase
private(set) var state = State()
private var cancellables = Set<AnyCancellable>()
init(
signInUseCase: SignInUseCase,
signOutUseCase: SignOutUseCase,
sessionUseCase: AuthSessionUseCase
) {
self.signInUseCase = signInUseCase
self.signOutUseCase = signOutUseCase
self.sessionUseCase = sessionUseCase
self.sessionUseCase.signedInPublisher
.removeDuplicates()
.receive(on: DispatchQueue.main)
.sink { [weak self] signIn in
self?.send(.setLogined(signIn))
}
.store(in: &cancellables)
}
func reduce(with action: Action) -> [SideEffect] {
var state = self.state
var effects: [SideEffect] = []
switch action {
case .setAlert(let isPresented):
setAlert(&state, isPresented: isPresented)
case .tapSignInButton(let authProvider):
effects = [.signIn(authProvider)]
case .tapSignOutButton, .signOutAuto:
effects = [.signOut]
case .setLoading(let value):
state.isLoading = value
case .setLogined(let result):
state.signIn = result
}
if self.state != state { self.state = state }
return effects
}
func run(_ effect: SideEffect) {
send(.setLoading(true))
switch effect {
case .signIn(let authProvider):
Task {
do {
defer { send(.setLoading(false)) }
try await self.signInUseCase.execute(authProvider)
send(.setLogined(true))
sessionUseCase.execute(true)
} catch {
send(.setLogined(false))
sessionUseCase.execute(false)
send(.setAlert(true))
}
}
case .signOut:
Task {
do {
defer { send(.setLoading(false)) }
try await self.signOutUseCase.execute()
send(.setLogined(false))
sessionUseCase.execute(false)
} catch {
send(.setAlert(true))
}
}
}
}
}
private extension LoginViewModel {
func setAlert(
_ state: inout State,
isPresented: Bool,
) {
state.alertTitle = "오류"
state.alertMessage = "문제가 발생했습니다. 잠시 후 다시 시도해주세요."
state.showAlert = isPresented
}
}