Files
e-party-iOS/yana/Utils/APILoading/APILoadingManager.swift
edwinQQQ 4a1b814902 feat: 实现数据迁移和用户信息管理优化
- 在AppDelegate中集成数据迁移管理器,支持从UserDefaults迁移到Keychain。
- 重构UserInfoManager,使用Keychain存储用户信息,增加内存缓存以提升性能。
- 添加API加载效果视图,增强用户体验。
- 更新SplashFeature以支持自动登录和认证状态检查。
- 语言设置迁移至Keychain,确保用户设置的安全性。
2025-07-10 17:20:20 +08:00

198 lines
6.0 KiB
Swift
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import Foundation
import SwiftUI
import Combine
// MARK: - API Loading Manager
/// API
///
///
/// - API
/// - loading
/// -
/// - 线
class APILoadingManager: ObservableObject {
// MARK: - Properties
///
static let shared = APILoadingManager()
///
@Published private(set) var loadingItems: [APILoadingItem] = []
///
private var errorCleanupTasks: [UUID: DispatchWorkItem] = [:]
///
private init() {}
// MARK: - Public Methods
/// loading
/// - Parameters:
/// - shouldShowLoading: loading
/// - shouldShowError:
/// - Returns: ID
func startLoading(shouldShowLoading: Bool = true, shouldShowError: Bool = true) -> UUID {
let loadingId = UUID()
let loadingItem = APILoadingItem(
id: loadingId,
state: .loading,
shouldShowError: shouldShowError,
shouldShowLoading: shouldShowLoading
)
// 🚨 线 @Published
DispatchQueue.main.async { [weak self] in
self?.loadingItems.append(loadingItem)
}
return loadingId
}
/// loading
/// - Parameter id: ID
func finishLoading(_ id: UUID) {
DispatchQueue.main.async { [weak self] in
self?.removeLoading(id)
}
}
/// loading
/// - Parameters:
/// - id: ID
/// - errorMessage:
func setError(_ id: UUID, errorMessage: String) {
DispatchQueue.main.async { [weak self] in
guard let self = self else { return }
//
if let index = self.loadingItems.firstIndex(where: { $0.id == id }) {
let currentItem = self.loadingItems[index]
//
if currentItem.shouldShowError {
let errorItem = APILoadingItem(
id: id,
state: .error(message: errorMessage),
shouldShowError: true,
shouldShowLoading: currentItem.shouldShowLoading
)
self.loadingItems[index] = errorItem
//
self.setupErrorCleanup(for: id)
} else {
//
self.loadingItems.removeAll { $0.id == id }
}
}
}
}
///
/// - Parameter id: ID
private func removeLoading(_ id: UUID) {
cancelErrorCleanup(for: id)
// 🚨 线 @Published
if Thread.isMainThread {
loadingItems.removeAll { $0.id == id }
} else {
DispatchQueue.main.async { [weak self] in
self?.loadingItems.removeAll { $0.id == id }
}
}
}
///
func clearAll() {
DispatchQueue.main.async { [weak self] in
guard let self = self else { return }
//
self.errorCleanupTasks.values.forEach { $0.cancel() }
self.errorCleanupTasks.removeAll()
//
self.loadingItems.removeAll()
}
}
// MARK: - Computed Properties
/// loading
var hasActiveLoading: Bool {
if Thread.isMainThread {
return loadingItems.contains { $0.state == .loading && $0.shouldDisplay }
} else {
return false
}
}
///
var hasActiveError: Bool {
if Thread.isMainThread {
return loadingItems.contains { $0.isError && $0.shouldDisplay }
} else {
return false
}
}
// MARK: - Private Methods
///
/// - Parameter id: ID
private func setupErrorCleanup(for id: UUID) {
let workItem = DispatchWorkItem { [weak self] in
self?.removeLoading(id)
}
errorCleanupTasks[id] = workItem
DispatchQueue.main.asyncAfter(
deadline: .now() + APILoadingConfiguration.errorDisplayDuration,
execute: workItem
)
}
///
/// - Parameter id: ID
private func cancelErrorCleanup(for id: UUID) {
errorCleanupTasks[id]?.cancel()
errorCleanupTasks.removeValue(forKey: id)
}
}
// MARK: - Convenience Extensions
extension APILoadingManager {
/// 便 loading
/// - Parameters:
/// - shouldShowLoading: loading
/// - shouldShowError:
/// - operation:
/// - Returns:
func withLoading<T>(
shouldShowLoading: Bool = true,
shouldShowError: Bool = true,
operation: @escaping () async throws -> T
) async -> Result<T, Error> {
let loadingId = startLoading(
shouldShowLoading: shouldShowLoading,
shouldShowError: shouldShowError
)
do {
let result = try await operation()
finishLoading(loadingId)
return .success(result)
} catch {
setError(loadingId, errorMessage: error.localizedDescription)
return .failure(error)
}
}
}