115 lines
3.1 KiB
Swift
115 lines
3.1 KiB
Swift
import Foundation
|
|
import SwiftUI
|
|
|
|
@MainActor
|
|
class AuthViewModel: ObservableObject {
|
|
@Published var isAuthenticated = false
|
|
@Published var currentUser: UserResponse?
|
|
@Published var isLoading = false
|
|
@Published var error: String?
|
|
|
|
private let auth = AuthService.shared
|
|
private let keychain = KeychainService.shared
|
|
|
|
func checkAuthStatus() async {
|
|
#if DEBUG
|
|
if PreviewData.isPreviewMode {
|
|
currentUser = PreviewData.mockUser
|
|
isAuthenticated = true
|
|
return
|
|
}
|
|
#endif
|
|
guard keychain.loadAccessToken() != nil else {
|
|
isAuthenticated = false
|
|
return
|
|
}
|
|
isLoading = true
|
|
defer { isLoading = false }
|
|
do {
|
|
currentUser = try await auth.getMe()
|
|
isAuthenticated = true
|
|
await requestPushIfNeeded()
|
|
} catch APIError.unauthorized {
|
|
isAuthenticated = false
|
|
} catch {
|
|
isAuthenticated = false
|
|
}
|
|
}
|
|
|
|
#if DEBUG
|
|
func enterPreviewMode() async {
|
|
PreviewData.isPreviewMode = true
|
|
currentUser = PreviewData.mockUser
|
|
isAuthenticated = true
|
|
await PreviewData.startMockLiveActivity()
|
|
}
|
|
#endif
|
|
|
|
func login(email: String, password: String) async {
|
|
isLoading = true
|
|
error = nil
|
|
defer { isLoading = false }
|
|
do {
|
|
currentUser = try await auth.login(email: email, password: password)
|
|
isAuthenticated = true
|
|
await requestPushIfNeeded()
|
|
} catch {
|
|
self.error = error.localizedDescription
|
|
}
|
|
}
|
|
|
|
func register(email: String, password: String) async -> Bool {
|
|
isLoading = true
|
|
error = nil
|
|
defer { isLoading = false }
|
|
do {
|
|
_ = try await auth.register(email: email, password: password)
|
|
return true
|
|
} catch {
|
|
self.error = error.localizedDescription
|
|
return false
|
|
}
|
|
}
|
|
|
|
func verifyEmail(email: String, code: String) async {
|
|
isLoading = true
|
|
error = nil
|
|
defer { isLoading = false }
|
|
do {
|
|
_ = try await auth.verifyEmail(email: email, code: code)
|
|
// Auto-login after verification is handled by calling login from view
|
|
} catch {
|
|
self.error = error.localizedDescription
|
|
}
|
|
}
|
|
|
|
func logout() async {
|
|
#if DEBUG
|
|
if PreviewData.isPreviewMode {
|
|
await PreviewData.stopMockLiveActivity()
|
|
PreviewData.isPreviewMode = false
|
|
isAuthenticated = false
|
|
currentUser = nil
|
|
return
|
|
}
|
|
#endif
|
|
isLoading = true
|
|
defer { isLoading = false }
|
|
do {
|
|
try await auth.logout()
|
|
} catch {
|
|
// Clear anyway
|
|
keychain.clearTokens()
|
|
}
|
|
isAuthenticated = false
|
|
currentUser = nil
|
|
}
|
|
|
|
private func requestPushIfNeeded() async {
|
|
let granted = await PushNotificationService.shared.requestPermission()
|
|
if granted {
|
|
PushNotificationService.shared.registerForRemoteNotifications()
|
|
}
|
|
}
|
|
}
|