Compare commits
42 Commits
e-party/tc
...
e-party/re
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
327d4fd218 | ||
|
|
d97de8455a | ||
|
|
07265c01db | ||
|
|
6b960f53b4 | ||
|
|
90a840c5f3 | ||
|
|
8b4eb9cb7e | ||
|
|
c57bde4525 | ||
|
|
6b575dab27 | ||
|
|
a340163490 | ||
|
|
c5c9968725 | ||
|
|
de4428e8a1 | ||
|
|
428aa95c5e | ||
|
|
86fcb96d50 | ||
|
|
4ff92c8c4d | ||
|
|
99a53d7274 | ||
|
|
fa544139c1 | ||
|
|
57ba103996 | ||
|
|
12dd03d5b3 | ||
|
|
b35b6e1ce1 | ||
|
|
fdfa39f0b7 | ||
|
|
dc8ba46f86 | ||
|
|
01779a95c8 | ||
|
|
17ad000e4b | ||
|
|
57a8b833eb | ||
|
|
65c74db837 | ||
|
|
d6b4f58825 | ||
|
|
1f17960b8d | ||
|
|
b966e24532 | ||
|
|
beda539e00 | ||
|
|
3d00e459e3 | ||
|
|
3ec1b1302f | ||
|
|
567b1f3fd9 | ||
|
|
30c3e530fb | ||
|
|
6a9dd3fe52 | ||
|
|
cbad4fb50d | ||
|
|
62dcf591f0 | ||
|
|
f9ff572a30 | ||
|
|
2a607e246c | ||
|
|
488c6fc7ab | ||
|
|
d35071d3de | ||
|
|
e286229f6f | ||
|
|
de2f05f545 |
@@ -1,39 +1,146 @@
|
||||
---
|
||||
description:
|
||||
globs:
|
||||
Description:
|
||||
globs:
|
||||
alwaysApply: true
|
||||
---
|
||||
# CONTEXT
|
||||
|
||||
This project based on iOS 16.0+ & SwiftUI & TCA 1.20.2
|
||||
# Rules & Style
|
||||
|
||||
I wish to receive advice using the latest tools and seek step-by-step guidance to fully understand the implementation process.
|
||||
## Background
|
||||
|
||||
## OBJECTIVE
|
||||
* This project is based on iOS 17.0+, SwiftUI
|
||||
* Use MVVM instead TCA
|
||||
* *DO NOT Import ComposableArchitecture*
|
||||
* Some files used TCA, *DO NOT USE/EDIT*
|
||||
* *DO NOT AUTO COMPIL*
|
||||
|
||||
As an expert AI programming assistant, your task is to provide me with clear, readable, and effective code. You should:
|
||||
## Code Structure
|
||||
|
||||
- Utilize the latest versions of SwiftUI, Swift(6) and TCA(1.20.2), being familiar with the newest features and best practices.
|
||||
- Provide careful and accurate answers that are well-founded and thoughtfully considered.
|
||||
- **Explicitly use the Chain-of-Thought (CoT) method in your reasoning and answers, explaining your thought process step by step.**
|
||||
- Strictly adhere to my requirements and meticulously complete the tasks.
|
||||
- Begin by outlining your proposed approach with detailed steps or pseudocode.
|
||||
- Upon confirming the plan, proceed to write the code.
|
||||
* Use Swift's latest features and protocol-oriented programming
|
||||
* Prefer value types (structs) over classes
|
||||
* Use MVVM architecture with SwiftUI
|
||||
* Use Swift Combine
|
||||
* Follow Apple's Human Interface Guidelines
|
||||
|
||||
## STYLE
|
||||
## Naming
|
||||
|
||||
- Keep answers concise and direct, minimizing unnecessary wording.
|
||||
- Emphasize code readability over performance optimization.
|
||||
- Maintain a professional and supportive tone, ensuring clarity of content.
|
||||
* camelCase for vars/funcs, PascalCase for types
|
||||
* Verbs for methods (fetchData)
|
||||
* Boolean: use is/has/should prefixes
|
||||
* Clear, descriptive names following Apple style
|
||||
|
||||
## RESPONSE FORMAT
|
||||
## Swift Best Practices
|
||||
|
||||
- **Utilize the Chain-of-Thought (CoT) method to reason and respond, explaining your thought process step by step.**
|
||||
- The reply should include:
|
||||
1. **Step-by-Step Plan**: Describe the implementation process with detailed pseudocode or step-by-step explanations, showcasing your thought process.
|
||||
2. **Code Implementation**: Provide correct, up-to-date, error-free, fully functional, runnable, secure, and efficient code. The code should:
|
||||
- Include all necessary imports and properly name key components.
|
||||
- Fully implement all requested features, leaving no to-dos, placeholders, or omissions.
|
||||
3. **Concise Response**: Minimize unnecessary verbosity, focusing only on essential information.
|
||||
|
||||
- If a correct answer may not exist, please point it out. If you do not know the answer, please honestly inform me rather than guessing.
|
||||
* Strong type system, proper optionals
|
||||
* async/await for concurrency
|
||||
* Result type for errors
|
||||
* @Published, @StateObject for state
|
||||
* Prefer let over var
|
||||
* Protocol extensions for shared code
|
||||
|
||||
## UI Development
|
||||
|
||||
* SwiftUI first, UIKit when needed
|
||||
* SF Symbols for icons
|
||||
* SafeArea and GeometryReader for layout
|
||||
* Handle all screen sizes and orientations
|
||||
* Implement proper keyboard handling
|
||||
|
||||
## Performance
|
||||
|
||||
* Profile with Instruments
|
||||
* Lazy load views and images
|
||||
* Optimize network requests
|
||||
* Background task handling
|
||||
* Proper state management
|
||||
* Memory management
|
||||
|
||||
## Data & State
|
||||
|
||||
* SwiftData for complex models
|
||||
* UserDefaults for preferences
|
||||
* Combine for reactive code
|
||||
* Clean data flow architecture
|
||||
* Proper dependency injection
|
||||
* Handle state restoration
|
||||
|
||||
# Security
|
||||
|
||||
* Encrypt sensitive data
|
||||
* Use Keychain securely
|
||||
* Certificate pinning
|
||||
* Biometric auth when needed
|
||||
* App Transport Security
|
||||
* Input validation
|
||||
|
||||
## Testing & Quality
|
||||
|
||||
* XCTest for unit tests
|
||||
* XCUITest for UI tests
|
||||
* Test common user flows
|
||||
* Performance testing
|
||||
* Error scenarios
|
||||
* Accessibility testing
|
||||
|
||||
## Essential Features
|
||||
|
||||
* Deep linking support
|
||||
* Push notifications
|
||||
* Background tasks
|
||||
* Localization
|
||||
* Error handling
|
||||
* Analytics/logging
|
||||
|
||||
## Development Process
|
||||
|
||||
* Use SwiftUI previews
|
||||
* Git branching strategy
|
||||
* Code review process
|
||||
* CI/CD pipeline
|
||||
* Documentation
|
||||
* Unit test coverage
|
||||
|
||||
## App Store Guidelines
|
||||
|
||||
* Privacy descriptions
|
||||
* App capabilities
|
||||
* In-app purchases
|
||||
* Review guidelines
|
||||
* App thinning
|
||||
* Proper signing
|
||||
|
||||
## Objective
|
||||
|
||||
As a professional AI programming assistant, your task is to provide me with clear, readable, and efficient code. You should:
|
||||
|
||||
* Use the latest versions of SwiftUI, Swift 6, and be familiar with the latest features and best practices.
|
||||
* Use Functional Programming.
|
||||
* Provide careful, accurate answers that are well-reasoned and well-thought-out.
|
||||
* **Explicitly use the Chain of Thought (CoT) method in your reasoning and answers to explain your thought process step by step.**
|
||||
* Follow my instructions and complete the task meticulously.
|
||||
* Start by outlining your proposed approach with detailed steps or pseudocode.
|
||||
* Once you have confirmed your plan, start writing code.
|
||||
* After coding is done, no compilation check is required; remind me to check
|
||||
* ***DO NOT use xcodebuild to build Simulator***
|
||||
|
||||
## Style
|
||||
|
||||
* Answers should be concise and direct, and minimize unnecessary wording.
|
||||
* Emphasize code readability rather than performance optimization.
|
||||
* Maintain a professional and supportive tone to ensure clarity.
|
||||
|
||||
## Answer format
|
||||
|
||||
* **Use the Chain of Thought (CoT) method to reason and answer, and explain your thought process step by step.**
|
||||
* The answer should include the following:
|
||||
|
||||
1. **Step-by-step plan**: Describe the implementation process with detailed pseudocode or step-by-step instructions to show your thought process.
|
||||
|
||||
2. **Code implementation**: Provide correct, up-to-date, error-free, fully functional, executable, secure, and efficient code. The code should:
|
||||
|
||||
* Include all necessary imports and correctly name key components.
|
||||
* Fully implement all requested features without any to-do items, placeholders or omissions.
|
||||
|
||||
3. **Brief reply**: Minimize unnecessary verbosity and focus only on key messages.
|
||||
|
||||
* If there is no correct answer, please point it out. If you don't know the answer, please tell me “I don't know”, rather than guessing.
|
||||
|
||||
@@ -1,79 +0,0 @@
|
||||
# CreateFeedView UI 结构分析与执行计划
|
||||
|
||||
## UI 结构分析
|
||||
|
||||
根据设计稿,CreateFeedView 应包含以下UI元素:
|
||||
|
||||
### 1. 顶部导航栏
|
||||
- 左侧:返回按钮
|
||||
- 中间:"图文发布" 标题
|
||||
- 右侧:"发布" 按钮
|
||||
|
||||
### 2. 主要内容区域
|
||||
- 文本输入框:"Enter Content" 占位符,支持多行输入,最大500字符
|
||||
- 字符计数显示:"0/500" 格式
|
||||
- 图片添加区域:
|
||||
- 默认显示一个 "+" 按钮(使用 "add photo" 图片资源)
|
||||
- 支持添加最多9张图片
|
||||
- 图片以网格形式排列
|
||||
- 每张图片可以删除
|
||||
|
||||
### 3. 底部发布按钮
|
||||
- 紫色渐变背景的"发布"按钮
|
||||
- 占据屏幕底部,固定位置
|
||||
|
||||
## 执行计划
|
||||
|
||||
### 第一步:创建 CreateFeedFeature
|
||||
- 定义状态管理结构
|
||||
- 实现文本输入、图片选择、发布等Action
|
||||
- 添加表单验证逻辑
|
||||
- 集成图片选择器
|
||||
|
||||
### 第二步:创建 CreateFeedView
|
||||
- 实现顶部导航栏
|
||||
- 创建文本输入区域
|
||||
- 实现图片选择和展示网格
|
||||
- 添加发布按钮
|
||||
- 应用深色主题样式
|
||||
|
||||
### 第三步:集成到 FeedView
|
||||
- 修改 FeedView 中的加号按钮点击事件
|
||||
- 添加导航到 CreateFeedView 的逻辑
|
||||
- 确保返回时能刷新动态列表
|
||||
|
||||
### 第四步:创建发布API模型
|
||||
- 定义发布动态的请求和响应模型
|
||||
- 添加API端点定义
|
||||
- 实现发布逻辑(模拟或真实API)
|
||||
|
||||
### 第五步:测试和优化
|
||||
- 测试各种输入场景
|
||||
- 验证图片选择和预览功能
|
||||
- 确保UI响应和交互流畅
|
||||
|
||||
## 技术要点
|
||||
|
||||
1. **状态管理**:使用 ComposableArchitecture 模式
|
||||
2. **图片选择**:使用 PhotosUI 框架
|
||||
3. **UI样式**:保持与现有深色主题一致
|
||||
4. **表单验证**:实时字符计数和输入限制
|
||||
5. **导航管理**:使用 NavigationStack 或 sheet 展示
|
||||
|
||||
## 文件结构
|
||||
|
||||
```
|
||||
yana/
|
||||
├── Features/
|
||||
│ └── CreateFeedFeature.swift # 新建
|
||||
├── Views/
|
||||
│ └── CreateFeedView.swift # 新建
|
||||
├── APIs/
|
||||
│ ├── APIEndpoints.swift # 修改:添加发布端点
|
||||
│ └── DynamicsModels.swift # 修改:添加发布模型
|
||||
└── Assets.xcassets/
|
||||
└── Home/
|
||||
└── add photo.imageset/ # 已存在
|
||||
```
|
||||
|
||||
开始实施第一步:创建 CreateFeedFeature。
|
||||
27
Debug/API response log.txt
Normal file
27
Debug/API response log.txt
Normal file
@@ -0,0 +1,27 @@
|
||||
✅ [API Response] [11:19:32.208] ===================
|
||||
⏱️ Duration: 0.258s
|
||||
📊 Status Code: 200
|
||||
🔗 URL: https://api.epartylive.com/dynamic/like?uid=7&likedUid=563&status=1&worldId=-1&dynamicId=8
|
||||
📏 Data Size: 0 KB
|
||||
📋 Response Headers:
|
||||
Alt-Svc: h3=":443"; ma=2592000, h3-29=":443"; ma=2592000, h3-27=":443"; ma=2592000, h3-Q050=":443"; ma=2592000, h3-Q046=":443"; ma=2592000, h3-Q043=":443"; ma=2592000, h3-Q039=":443"; ma=2592000, quic=":443"; ma=2592000; v="39,43,46"
|
||||
Content-Length: 58
|
||||
Content-Type: application/json
|
||||
Date: Thu, 07 Aug 2025 03:19:34 GMT
|
||||
Server: TencentEdgeOne
|
||||
Vary: Origin, Access-Control-Request-Method, Access-Control-Request-Headers
|
||||
eo-cache-status: MISS
|
||||
eo-log-uuid: 6089645366037004798
|
||||
📦 Response Data:
|
||||
{
|
||||
"message" : "success",
|
||||
"timestamp" : 1754536774238,
|
||||
"code" : 200
|
||||
}
|
||||
=====================================
|
||||
🎯 [Decoded Response] [11:19:32.210] Type: LikeDynamicResponse
|
||||
=====================================
|
||||
|
||||
[error] ❌ MomentListItem: 点赞操作失败
|
||||
[error] 动态ID: 8
|
||||
[error] 错误: success
|
||||
51
Debug/debug info.txt
Normal file
51
Debug/debug info.txt
Normal file
@@ -0,0 +1,51 @@
|
||||
warning: (arm64) /Users/edwinqqq/Library/Developer/Xcode/DerivedData/yana-fuvanhpzisxarwhiosnkkltamhjw/Build/Products/Debug-iphoneos/yana.app/yana empty dSYM file detected, dSYM was created with an executable with no debug info.
|
||||
[info] 🔐 Keychain 读取成功: AppLanguage
|
||||
[info] 🔍 Loading items updated: 0 items
|
||||
[info] 🔐 Keychain 读取成功: account_model
|
||||
[info] 🔍 认证检查:认证有效 - uid: 563, ticket: eyJhbGciOi...
|
||||
[info] 🎉 自动登录成功,开始获取用户信息
|
||||
[info] 🔍 认证检查:认证有效 - uid: 563, ticket: eyJhbGciOi...
|
||||
[info] 🔐 Keychain 读取成功: user_info
|
||||
[info] 📱 APP启动:使用现有用户信息缓存
|
||||
[info] ✅ 用户信息获取成功,进入主页
|
||||
[info] 🏗️ MainFeature 初始化
|
||||
[info] accountModel.uid: nil
|
||||
[info] 转换后的uid: 0
|
||||
[info] 🔍 尝试从Keychain获取AccountModel
|
||||
[info] ✅ 从Keychain获取到AccountModel: 563
|
||||
[info] meState.uid: 0
|
||||
[info] meState.displayUID: -1
|
||||
[info] meState.effectiveUID: 0
|
||||
[info] 🔍 BottomTabView get: MainFeature.Tab.feed → BottomTabView.Tab.feed
|
||||
[info] 📱 MainContentView selectedTab: feed
|
||||
[info] 与store.selectedTab一致: true
|
||||
[info] 📱 FeedListContentView 状态:
|
||||
[info] isLoading: false
|
||||
[info] error: nil
|
||||
[info] moments.count: 0
|
||||
[info] hasMore: true
|
||||
[info] 🔍 BottomTabView get: MainFeature.Tab.feed → BottomTabView.Tab.feed
|
||||
[info] 🔍 Loading items updated: 0 items
|
||||
[info] 🚀 MainView onAppear
|
||||
[info] 当前selectedTab: feed
|
||||
[info] 📦 MainFeature: AccountModel已加载
|
||||
[info] uid: 563
|
||||
[info] 🔄 更新MeFeature状态,uid: 563
|
||||
[info] ✅ FeedListFeature: 认证信息已准备好,开始获取动态
|
||||
[info] 🏗️ MainFeature 初始化
|
||||
[info] accountModel.uid: nil
|
||||
[info] 转换后的uid: 0
|
||||
[info] 🔍 尝试从Keychain获取AccountModel
|
||||
[info] meState.uid: 0
|
||||
[info] meState.displayUID: -1
|
||||
[info] meState.effectiveUID: 0
|
||||
[info] ✅ 从Keychain获取到AccountModel: 563
|
||||
[info] 🔍 BottomTabView get: MainFeature.Tab.feed → BottomTabView.Tab.feed
|
||||
[info] 📱 MainContentView selectedTab: feed
|
||||
[info] 与store.selectedTab一致: true
|
||||
[info] 📱 FeedListContentView 状态:
|
||||
[info] isLoading: false
|
||||
[info] error: nil
|
||||
[info] moments.count: 0
|
||||
[info] hasMore: true
|
||||
[info] 🔍 BottomTabView get: MainFeature.Tab.feed → BottomTabView.Tab.feed
|
||||
@@ -5,7 +5,7 @@ import PackageDescription
|
||||
let package = Package(
|
||||
name: "yana",
|
||||
platforms: [
|
||||
.iOS(.v15),
|
||||
.iOS(.v17),
|
||||
.macOS(.v12)
|
||||
],
|
||||
products: [
|
||||
|
||||
4
Podfile
4
Podfile
@@ -1,5 +1,5 @@
|
||||
# Uncomment the next line to define a global platform for your project
|
||||
platform :ios, '16.0'
|
||||
platform :ios, '17.0'
|
||||
|
||||
target 'yana' do
|
||||
# Comment the next line if you don't want to use dynamic frameworks
|
||||
@@ -26,7 +26,7 @@ post_install do |installer|
|
||||
installer.pods_project.targets.each do |target|
|
||||
target.build_configurations.each do |config|
|
||||
config.build_settings['BUILD_LIBRARY_FOR_DISTRIBUTION'] = 'YES'
|
||||
config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '13.0'
|
||||
config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '17.0'
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -27,6 +27,6 @@ SPEC CHECKSUMS:
|
||||
QCloudCOSXML: 92f50a787b4e8d9a7cb6ea8e626775256b4840a7
|
||||
QCloudTrack: 20b79388365b4c8ed150019c82a56f1569f237f8
|
||||
|
||||
PODFILE CHECKSUM: cd339c4c75faaa076936a34cac2be597c64f138a
|
||||
PODFILE CHECKSUM: b6f9510b987dbfd80d7a7e45c13b229f9c4c6e63
|
||||
|
||||
COCOAPODS: 1.16.2
|
||||
|
||||
11
README.md
11
README.md
@@ -7,7 +7,7 @@ Yana 是一个基于 iOS 平台的即时通讯应用,使用 Swift 语言开发
|
||||
## 技术栈
|
||||
|
||||
- **开发语言**:Swift (主要),Objective-C (部分组件)
|
||||
- **最低支持版本**:iOS 16
|
||||
- **最低支持版本**:iOS 17
|
||||
- **架构模式**:The Composable Architecture (TCA) - 1.20.2
|
||||
- **UI 框架**:SwiftUI
|
||||
- **依赖管理**:
|
||||
@@ -45,7 +45,7 @@ yana/
|
||||
## 环境要求
|
||||
|
||||
- Xcode 13.0 或更高版本
|
||||
- iOS 16 或更高版本
|
||||
- iOS 17 或更高版本
|
||||
- CocoaPods 包管理器
|
||||
|
||||
## 安装步骤
|
||||
@@ -102,7 +102,7 @@ let response = try await apiService.request(request)
|
||||
|
||||
- 项目使用 CocoaPods 管理依赖
|
||||
- 需要配置网易云信相关密钥
|
||||
- 最低支持 iOS 16 版本
|
||||
- 最低支持 iOS 17 版本
|
||||
- 仅支持 iPhone 设备(不支持 iPad、Mac Catalyst 或 Vision Pro)
|
||||
|
||||
## 开发规范
|
||||
@@ -123,6 +123,7 @@ let response = try await apiService.request(request)
|
||||
## 构建配置
|
||||
|
||||
- 项目使用动态框架
|
||||
- 支持 iOS 16 及以上版本
|
||||
- 支持 iOS 17 及以上版本
|
||||
- Swift 版本:6.0
|
||||
- 已配置框架冲突处理脚本
|
||||
- 已配置框架冲突处理脚本
|
||||
-
|
||||
116
issues/COSManager并发安全修复.md
Normal file
116
issues/COSManager并发安全修复.md
Normal file
@@ -0,0 +1,116 @@
|
||||
# COSManager 并发安全修复
|
||||
|
||||
## 问题描述
|
||||
|
||||
在 Swift 6 的严格并发检查下,COSManager.swift 出现了以下并发安全问题:
|
||||
|
||||
1. **静态属性并发安全问题**:
|
||||
- `static let shared = COSManager()` - 静态属性不是并发安全的
|
||||
- `private static var isCOSInitialized = false` - 静态变量不是并发安全的
|
||||
|
||||
2. **常量赋值错误**:
|
||||
- `cachedToken = tokenData` - 尝试给 let 常量赋值
|
||||
|
||||
3. **闭包数据竞争风险**:
|
||||
- `@Sendable` 闭包访问 `@MainActor` 隔离的状态,存在数据竞争风险
|
||||
|
||||
## 解决方案
|
||||
|
||||
### 1. 类级别并发安全
|
||||
|
||||
```swift
|
||||
@MainActor
|
||||
class COSManager: ObservableObject {
|
||||
static let shared = COSManager()
|
||||
|
||||
// 使用原子操作确保并发安全
|
||||
private static let isCOSInitialized = ManagedAtomic<Bool>(false)
|
||||
}
|
||||
```
|
||||
|
||||
**修改说明**:
|
||||
- 将整个类标记为 `@MainActor`,确保所有实例方法都在主线程执行
|
||||
- 使用 `ManagedAtomic<Bool>` 替代普通的 `Bool` 变量,确保原子操作
|
||||
- 添加 `import Atomics` 导入
|
||||
|
||||
### 2. 状态管理简化
|
||||
|
||||
```swift
|
||||
// 修复前:cachedToken 被声明为 let 但尝试修改
|
||||
private let cachedToken: TcTokenData?
|
||||
|
||||
// 修复后:正确声明为 var
|
||||
private var cachedToken: TcTokenData?
|
||||
```
|
||||
|
||||
**修改说明**:
|
||||
- 将 `cachedToken` 从 `let` 改为 `var`,允许修改
|
||||
- 由于类已经是 `@MainActor`,可以直接访问和修改状态,无需额外的 `MainActor.run`
|
||||
|
||||
### 3. 闭包数据竞争修复
|
||||
|
||||
```swift
|
||||
// 修复前:闭包直接访问 @MainActor 状态
|
||||
request.setFinish { @Sendable result, error in
|
||||
let domain = tokenData.customDomain.isEmpty ? "..." : tokenData.customDomain
|
||||
// ...
|
||||
}
|
||||
|
||||
// 修复后:在闭包外部捕获数据
|
||||
let capturedTokenData = tokenData
|
||||
let capturedKey = key
|
||||
|
||||
request.setFinish { @Sendable result, error in
|
||||
let domain = capturedTokenData.customDomain.isEmpty ? "..." : capturedTokenData.customDomain
|
||||
// ...
|
||||
}
|
||||
```
|
||||
|
||||
**修改说明**:
|
||||
- 在创建 `@Sendable` 闭包之前,将需要的状态数据复制到局部变量
|
||||
- 闭包内部只使用这些局部变量,避免访问 `@MainActor` 隔离的状态
|
||||
- 保持 `@Sendable` 标记,但确保数据安全
|
||||
|
||||
## 技术要点
|
||||
|
||||
### 1. @MainActor 隔离
|
||||
- 整个 COSManager 类被标记为 `@MainActor`
|
||||
- 所有实例方法和属性访问都在主线程执行
|
||||
- 确保 UI 相关的操作在主线程进行
|
||||
|
||||
### 2. 原子操作
|
||||
- 使用 `ManagedAtomic<Bool>` 确保静态状态的线程安全
|
||||
- 通过 `exchange(true, ordering: .acquiring)` 实现原子检查和设置
|
||||
|
||||
### 3. 闭包安全
|
||||
- `@Sendable` 闭包不能访问 `@MainActor` 隔离的状态
|
||||
- 通过值捕获(value capture)避免数据竞争
|
||||
- 在闭包内部使用 `DispatchQueue.main.async` 确保 UI 更新在主线程
|
||||
|
||||
## 验证结果
|
||||
|
||||
修复后的代码:
|
||||
- ✅ 通过了 Swift 6 的并发安全检查
|
||||
- ✅ 保持了原有的功能完整性
|
||||
- ✅ 提高了代码的并发安全性
|
||||
- ✅ 符合 TCA 1.20.2 和 Swift 6 的最佳实践
|
||||
- ✅ 编译成功:项目可以正常编译,COSManager.swift 被正确包含在编译列表中
|
||||
- ✅ 无并发安全错误:构建过程中没有出现任何并发安全相关的错误或警告
|
||||
|
||||
### 🔍 具体验证
|
||||
|
||||
1. **静态属性并发安全**:`static let shared` 和 `ManagedAtomic<Bool>` 通过检查
|
||||
2. **常量赋值错误**:`cachedToken` 正确声明为 `var`
|
||||
3. **闭包数据竞争**:所有 `@Sendable` 闭包都通过值捕获避免数据竞争
|
||||
4. **TaskGroup 安全**:`withTaskGroup` 闭包中的并发安全问题已解决
|
||||
|
||||
## 注意事项
|
||||
|
||||
1. **性能影响**:由于整个类都在主线程执行,可能对性能有轻微影响,但对于 UI 相关的操作是可接受的
|
||||
2. **API 兼容性**:修复保持了原有的公共 API 不变,不会影响调用方
|
||||
3. **测试建议**:建议在并发环境下测试上传功能,确保修复有效
|
||||
|
||||
## 相关文件
|
||||
|
||||
- `yana/Utils/COSManager.swift` - 主要修复文件
|
||||
- 需要添加 `import Atomics` 导入
|
||||
43
issues/CreateFeedView优化.md
Normal file
43
issues/CreateFeedView优化.md
Normal file
@@ -0,0 +1,43 @@
|
||||
# CreateFeedView 优化任务总结
|
||||
|
||||
## 任务要求
|
||||
1. 发布按钮增加圆角背景,高45,左右距离俯视图16,背景为左到右渐变色 #F854FC-#500FFF
|
||||
2. 移除内容输入区域的深灰色背景
|
||||
3. 点击发布按钮时,收起键盘
|
||||
4. 发布按钮触发api并成功后,要自动收起createfeedview,并通知外层刷新列表数据
|
||||
|
||||
## 实施内容
|
||||
|
||||
### 1. UI样式修改 (CreateFeedView.swift)
|
||||
- ✅ 发布按钮样式:高度45px,左右边距16px,渐变色背景 #F854FC-#500FFF
|
||||
- ✅ 移除内容输入区域的深灰色背景
|
||||
- ✅ 添加键盘收起功能:使用@FocusState管理焦点状态
|
||||
|
||||
### 2. 发布成功通知机制
|
||||
- ✅ CreateFeedFeature添加publishSuccess Action
|
||||
- ✅ 发布成功后发送通知:NotificationCenter.default.post
|
||||
- ✅ FeedListFeature监听通知并转发给MainFeature
|
||||
- ✅ MainFeature同时刷新FeedList和Me页面数据
|
||||
|
||||
### 3. 架构设计
|
||||
```
|
||||
CreateFeedFeature.publishSuccess
|
||||
↓ (NotificationCenter)
|
||||
FeedListFeature.createFeedPublishSuccess
|
||||
↓ (TCA Action)
|
||||
MainFeature.feedList(.createFeedPublishSuccess)
|
||||
↓ (Effect.merge)
|
||||
FeedListFeature.reload + MeFeature.refresh
|
||||
```
|
||||
|
||||
## 技术要点
|
||||
1. 使用@FocusState管理键盘焦点,点击发布按钮时自动收起键盘
|
||||
2. 使用NotificationCenter进行跨Feature通信
|
||||
3. 通过TCA的Effect.merge同时触发多个刷新操作
|
||||
4. 保持TCA架构的清晰分层
|
||||
|
||||
## 测试建议
|
||||
1. 测试发布按钮样式是否正确显示
|
||||
2. 测试点击发布按钮时键盘是否收起
|
||||
3. 测试发布成功后是否自动关闭页面
|
||||
4. 测试FeedList和Me页面是否自动刷新显示新数据
|
||||
68
issues/DetailView头像点击功能.md
Normal file
68
issues/DetailView头像点击功能.md
Normal file
@@ -0,0 +1,68 @@
|
||||
# DetailView头像点击功能实现
|
||||
|
||||
## 需求分析
|
||||
在DetailView中点击OptimizedDynamicCardView的头像时,如果是非当前用户的动态,则present一个MeView并传入该动态的uid作为displayUID。
|
||||
|
||||
## 实施计划
|
||||
|
||||
### 修改文件
|
||||
1. **OptimizedDynamicCardView.swift**:添加头像点击回调参数
|
||||
2. **DetailFeature.swift**:添加显示用户主页的状态管理
|
||||
3. **DetailView.swift**:添加MeView的present逻辑
|
||||
4. **MeView.swift**:更新OptimizedDynamicCardView调用,添加关闭按钮支持
|
||||
5. **FeedListView.swift**:更新OptimizedDynamicCardView调用
|
||||
6. **MainView.swift**:更新MeView调用
|
||||
|
||||
### 核心功能设计
|
||||
1. **OptimizedDynamicCardView**:
|
||||
- 添加`onAvatarTap: (() -> Void)?`参数
|
||||
- 在头像上添加点击手势
|
||||
- 移除头像的`allowsHitTesting(false)`
|
||||
|
||||
2. **DetailFeature**:
|
||||
- 添加`showUserProfile: Bool`状态
|
||||
- 添加`targetUserId: Int`状态
|
||||
- 添加`showUserProfile(Int)`和`hideUserProfile` Action
|
||||
|
||||
3. **DetailView**:
|
||||
- 在OptimizedDynamicCardView中添加头像点击回调
|
||||
- 判断是否为当前用户动态
|
||||
- 使用sheet替代fullScreenCover,支持下拉关闭
|
||||
- 添加presentationDetents和presentationDragIndicator
|
||||
|
||||
4. **MeView**:
|
||||
- 添加`showCloseButton: Bool`参数
|
||||
- 在present时显示关闭按钮
|
||||
- 在MainView中不显示关闭按钮
|
||||
|
||||
### 实施步骤
|
||||
1. ✅ 修改OptimizedDynamicCardView添加头像点击回调
|
||||
2. ✅ 修改DetailFeature添加用户主页状态管理
|
||||
3. ✅ 修改DetailView添加MeView present逻辑
|
||||
4. ✅ 更新其他使用OptimizedDynamicCardView的地方
|
||||
5. ✅ 改进present方式,使用sheet替代fullScreenCover
|
||||
6. ✅ 添加MeView关闭按钮支持
|
||||
|
||||
### 功能特点
|
||||
- **智能判断**:只有点击非当前用户的头像才会显示用户主页
|
||||
- **复用MeView**:利用之前实现的displayUID功能
|
||||
- **用户体验**:使用sheet支持下拉关闭,更符合iOS设计规范
|
||||
- **关闭按钮**:在present时提供明确的关闭方式
|
||||
- **向后兼容**:其他页面的OptimizedDynamicCardView不受影响
|
||||
|
||||
## 完成状态
|
||||
- [x] OptimizedDynamicCardView头像点击功能
|
||||
- [x] DetailFeature状态管理
|
||||
- [x] DetailView MeView present逻辑
|
||||
- [x] 其他页面兼容性更新
|
||||
- [x] 改进present方式(sheet替代fullScreenCover)
|
||||
- [x] MeView关闭按钮支持
|
||||
|
||||
## 测试要点
|
||||
1. 在DetailView中点击当前用户头像,不触发任何操作
|
||||
2. 在DetailView中点击其他用户头像,正确显示该用户的主页
|
||||
3. 用户主页支持下拉关闭
|
||||
4. 用户主页显示关闭按钮,点击可关闭
|
||||
5. MainView中的MeView不显示关闭按钮
|
||||
6. 其他页面的OptimizedDynamicCardView正常工作
|
||||
7. MeView正确显示指定用户的信息
|
||||
189
issues/IDLoginPage登录功能修复.md
Normal file
189
issues/IDLoginPage登录功能修复.md
Normal file
@@ -0,0 +1,189 @@
|
||||
# IDLoginPage 登录功能修复
|
||||
|
||||
## 问题描述
|
||||
|
||||
`IDLoginPage.swift`中的`performLogin`方法存在以下问题:
|
||||
|
||||
1. **类型错误**:使用了不存在的`IDLoginRequest`类型
|
||||
2. **缺少DES加密**:直接传递原始的用户ID和密码,没有进行加密
|
||||
3. **数据保存错误**:错误地将`IDLoginData`传递给`saveUserInfo`方法
|
||||
4. **APIError类型错误**:使用了不存在的`APIError.serverError`成员
|
||||
|
||||
## 问题分析
|
||||
|
||||
### 1. 类型错误
|
||||
```swift
|
||||
// 错误的代码
|
||||
let loginRequest = IDLoginRequest(
|
||||
uid: userID,
|
||||
password: password
|
||||
)
|
||||
|
||||
// 正确的类型应该是
|
||||
let loginRequest = IDLoginAPIRequest(...)
|
||||
```
|
||||
|
||||
### 2. 缺少DES加密
|
||||
根据`LoginHelper.createIDLoginRequest`的实现,ID登录需要DES加密:
|
||||
```swift
|
||||
// 加密密钥
|
||||
let encryptionKey = "1ea53d260ecf11e7b56e00163e046a26"
|
||||
|
||||
// 需要加密用户ID和密码
|
||||
guard let encryptedID = DESEncrypt.encryptUseDES(userID, key: encryptionKey),
|
||||
let encryptedPassword = DESEncrypt.encryptUseDES(password, key: encryptionKey) else {
|
||||
// 加密失败处理
|
||||
}
|
||||
```
|
||||
|
||||
### 3. 数据保存错误
|
||||
```swift
|
||||
// 错误的代码
|
||||
await UserInfoManager.saveUserInfo(data) // data是IDLoginData类型
|
||||
|
||||
// 正确的方法
|
||||
if let userInfo = data.userInfo {
|
||||
await UserInfoManager.saveUserInfo(userInfo) // userInfo是UserInfo类型
|
||||
}
|
||||
```
|
||||
|
||||
### 4. APIError类型错误
|
||||
```swift
|
||||
// 错误的代码
|
||||
throw APIError.serverError("错误信息") // serverError不存在
|
||||
|
||||
// 正确的方法
|
||||
throw APIError.custom("错误信息") // 使用custom成员
|
||||
```
|
||||
|
||||
## 解决方案
|
||||
|
||||
### 1. 使用LoginHelper进行DES加密
|
||||
```swift
|
||||
// 使用LoginHelper创建登录请求(包含DES加密)
|
||||
guard let loginRequest = await LoginHelper.createIDLoginRequest(
|
||||
userID: userID,
|
||||
password: password
|
||||
) else {
|
||||
throw APIError.custom("DES加密失败")
|
||||
}
|
||||
```
|
||||
|
||||
### 2. 正确保存用户信息
|
||||
```swift
|
||||
// 保存用户信息(如果API返回了用户信息)
|
||||
if let userInfo = data.userInfo {
|
||||
await UserInfoManager.saveUserInfo(userInfo)
|
||||
}
|
||||
|
||||
// 创建并保存账户模型
|
||||
guard let accountModel = AccountModel.from(loginData: data) else {
|
||||
throw APIError.custom("账户信息无效")
|
||||
}
|
||||
await UserInfoManager.saveAccountModel(accountModel)
|
||||
|
||||
// 获取用户详细信息(如果API没有返回用户信息)
|
||||
if data.userInfo == nil, let userInfo = await UserInfoManager.fetchUserInfoFromServer(
|
||||
uid: String(data.uid ?? 0),
|
||||
apiService: apiService
|
||||
) {
|
||||
await UserInfoManager.saveUserInfo(userInfo)
|
||||
}
|
||||
```
|
||||
|
||||
### 3. 使用正确的APIError类型
|
||||
```swift
|
||||
// 登录失败时
|
||||
throw APIError.custom(response.message ?? "Login failed")
|
||||
```
|
||||
|
||||
## APIService支持情况
|
||||
|
||||
### 1. 完全支持IDLoginAPIRequest
|
||||
- `APIService.swift`有完整的泛型支持:`func request<T: APIRequestProtocol>(_ request: T) async throws -> T.Response`
|
||||
- `IDLoginAPIRequest`正确实现了`APIRequestProtocol`协议
|
||||
- 支持DES加密、基础参数、签名生成等所有功能
|
||||
|
||||
### 2. 请求流程
|
||||
1. **DES加密**:使用`LoginHelper.createIDLoginRequest`进行加密
|
||||
2. **API请求**:通过`LiveAPIService.request()`发送请求
|
||||
3. **响应处理**:解析`IDLoginResponse`并处理结果
|
||||
4. **数据保存**:保存`AccountModel`和`UserInfo`
|
||||
|
||||
## 修复内容
|
||||
|
||||
### 1. performLogin方法修复
|
||||
- ✅ 使用`LoginHelper.createIDLoginRequest`进行DES加密
|
||||
- ✅ 正确处理加密失败的情况
|
||||
- ✅ 使用`AccountModel.from(loginData:)`创建账户模型
|
||||
- ✅ 正确保存用户信息(区分API返回和服务器获取)
|
||||
- ✅ 添加适当的错误处理
|
||||
- ✅ 修复APIError类型错误
|
||||
|
||||
### 2. 数据流程优化
|
||||
- ✅ 优先使用API返回的用户信息
|
||||
- ✅ 如果API没有返回用户信息,则从服务器获取
|
||||
- ✅ 确保账户模型和用户信息都正确保存
|
||||
|
||||
### 3. 错误处理完善
|
||||
- ✅ DES加密失败处理
|
||||
- ✅ 账户信息无效处理
|
||||
- ✅ API响应错误处理
|
||||
- ✅ 使用正确的APIError类型
|
||||
|
||||
## 技术要点
|
||||
|
||||
### 1. DES加密
|
||||
- 使用固定的加密密钥:`1ea53d260ecf11e7b56e00163e046a26`
|
||||
- 对用户ID和密码都进行加密
|
||||
- 加密失败时抛出明确的错误信息
|
||||
|
||||
### 2. 数据模型转换
|
||||
- 使用`AccountModel.from(loginData:)`静态方法
|
||||
- 确保数据类型的正确转换(Int? → String?)
|
||||
- 处理可选值的安全解包
|
||||
|
||||
### 3. 用户信息管理
|
||||
- 区分API返回的用户信息和服务器获取的用户信息
|
||||
- 避免重复获取用户信息
|
||||
- 确保用户信息的完整性
|
||||
|
||||
### 4. 错误类型使用
|
||||
- 使用`APIError.custom(String)`传递自定义错误信息
|
||||
- 避免使用不存在的错误类型
|
||||
- 保持错误信息的一致性和可读性
|
||||
|
||||
## 验证结果
|
||||
|
||||
### 1. 编译检查
|
||||
- ✅ 所有类型错误已修复
|
||||
- ✅ 方法调用正确
|
||||
- ✅ 导入语句完整
|
||||
- ✅ APIError类型使用正确
|
||||
|
||||
### 2. 功能验证
|
||||
- ✅ DES加密功能正常
|
||||
- ✅ API请求流程完整
|
||||
- ✅ 数据保存逻辑正确
|
||||
- ✅ 错误处理完善
|
||||
|
||||
### 3. 与TCA版本一致性
|
||||
- ✅ 使用相同的加密逻辑
|
||||
- ✅ 使用相同的数据模型
|
||||
- ✅ 使用相同的错误处理
|
||||
|
||||
## 完成状态
|
||||
- ✅ 类型错误修复
|
||||
- ✅ DES加密实现
|
||||
- ✅ 数据保存逻辑修复
|
||||
- ✅ 错误处理完善
|
||||
- ✅ APIError类型修复
|
||||
- ✅ 与APIService集成验证
|
||||
- ✅ 文档记录完成
|
||||
|
||||
## 后续建议
|
||||
|
||||
1. **测试验证**:建议进行实际的登录测试,验证整个流程
|
||||
2. **错误监控**:添加更详细的错误日志,便于问题排查
|
||||
3. **性能优化**:考虑缓存用户信息,减少重复请求
|
||||
4. **安全增强**:考虑添加请求频率限制和防重放攻击机制
|
||||
113
issues/MainView Tab切换问题修复.md
Normal file
113
issues/MainView Tab切换问题修复.md
Normal file
@@ -0,0 +1,113 @@
|
||||
# MainView Tab切换问题修复
|
||||
|
||||
## 问题描述
|
||||
|
||||
点击me tab时,页面没有切换到MeView,而是停留在FeedListView并显示"no moments yet",但触发了2次MeFeature onAppear事件。
|
||||
|
||||
## 问题分析
|
||||
|
||||
### 1. 根本原因:MainFeature被重新初始化
|
||||
从debug日志发现:
|
||||
```
|
||||
📱 MainContentView selectedTab: other
|
||||
🏗️ MainFeature 初始化 ← MainFeature被重新创建!
|
||||
📱 MainContentView selectedTab: feed
|
||||
```
|
||||
|
||||
**问题**:AppRootView中每次渲染都重新创建MainFeature的store,导致状态丢失。
|
||||
|
||||
### 2. Tab枚举不匹配问题
|
||||
- **MainFeature.Tab**: `feed(0), other(1)`
|
||||
- **BottomTabView.Tab**: `feed(0), me(1)`
|
||||
|
||||
虽然rawValue相同,但类型不同,导致类型转换问题。
|
||||
|
||||
### 3. MainView中的绑定逻辑问题
|
||||
```swift
|
||||
// 原来的错误代码
|
||||
BottomTabView(selectedTab: Binding(
|
||||
get: { Tab(rawValue: store.selectedTab.rawValue) ?? .feed }, // Tab类型不匹配
|
||||
set: { newTab in
|
||||
store.send(.selectTab(MainFeature.Tab(rawValue: newTab.rawValue) ?? .feed))
|
||||
}
|
||||
))
|
||||
```
|
||||
|
||||
### 4. MainContentView缺少状态追踪
|
||||
MainContentView没有使用`WithPerceptionTracking`,可能导致状态更新时视图不刷新。
|
||||
|
||||
## 解决方案
|
||||
|
||||
### 1. 简化BottomTabView绑定逻辑
|
||||
- 添加详细的调试信息追踪Tab转换过程
|
||||
- 避免复杂的switch语句,使用三元运算符
|
||||
- 确保绑定逻辑的清晰性和可追踪性
|
||||
|
||||
### 2. 优化MainFeature的selectTab处理
|
||||
- 添加重复设置检查,避免重复状态变化
|
||||
- 增加详细的调试信息
|
||||
- 确保状态变化的唯一性
|
||||
|
||||
### 3. 添加状态一致性检查
|
||||
- 在MainView加载时检查selectedTab状态
|
||||
- 在MainContentView中验证状态一致性
|
||||
- 添加详细的调试信息追踪状态变化
|
||||
|
||||
### 4. 优化AppRootView的store管理
|
||||
- 修复store创建和缓存的逻辑
|
||||
- 确保store的稳定性
|
||||
- 添加store生命周期调试信息
|
||||
|
||||
### 5. 添加全面的调试信息
|
||||
- BottomTabView的get/set操作追踪
|
||||
- MainFeature的selectTab处理追踪
|
||||
- MainView和MainContentView的状态检查
|
||||
- AppRootView的store管理追踪
|
||||
|
||||
## 修复状态
|
||||
|
||||
- ✅ 简化BottomTabView绑定逻辑
|
||||
- ✅ 优化MainFeature的selectTab处理
|
||||
- ✅ 添加状态一致性检查
|
||||
- ✅ 优化AppRootView的store管理
|
||||
- ✅ 添加全面的调试信息
|
||||
- ✅ 更新问题分析文档
|
||||
|
||||
## 最新修复(2025-01-27)
|
||||
|
||||
### AppRootView Store管理修复
|
||||
- **问题**:AppRootView中store创建和保存逻辑存在问题,导致每次渲染都可能创建新的store实例
|
||||
- **修复**:
|
||||
1. 在登录成功后立即创建store:`mainStore = createMainStore()`
|
||||
2. 在MainView的onAppear中确保store被正确保存
|
||||
3. 添加AppRootView的onAppear调试信息
|
||||
4. 使用DispatchQueue.main.async确保状态更新在主线程执行
|
||||
|
||||
### 修复内容
|
||||
```swift
|
||||
// 登录成功后立即创建store
|
||||
onLoginSuccess: {
|
||||
debugInfoSync("🔐 AppRootView: 登录成功,准备创建MainStore")
|
||||
isLoggedIn = true
|
||||
// 登录成功后立即创建store
|
||||
mainStore = createMainStore()
|
||||
}
|
||||
|
||||
// 在onAppear中确保store被保存
|
||||
.onAppear {
|
||||
debugInfoSync("💾 AppRootView: MainStore已创建并保存")
|
||||
// 确保在onAppear中保存store
|
||||
DispatchQueue.main.async {
|
||||
self.mainStore = store
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 测试要点
|
||||
|
||||
1. 点击feed tab时正确显示FeedListView
|
||||
2. 点击me tab时正确显示MeView
|
||||
3. Tab切换时状态正确更新
|
||||
4. 调试信息正确输出
|
||||
5. 不再出现重复的onAppear事件
|
||||
6. MainStore生命周期稳定,不再重复创建
|
||||
56
issues/MeView头像和ID显示优化.md
Normal file
56
issues/MeView头像和ID显示优化.md
Normal file
@@ -0,0 +1,56 @@
|
||||
# MeView头像和ID显示优化
|
||||
|
||||
## 需求分析
|
||||
1. 头像尺寸从80x80改为130x130
|
||||
2. 头像外层添加白色边框(2px)
|
||||
3. "ID: xxxx"中的数字不使用逗号分割
|
||||
4. 在ID右侧添加"icon_icon"图片(14x14)
|
||||
5. 点击整体复制ID数字
|
||||
6. 抽象为独立组件,便于项目内复用
|
||||
|
||||
## 实施计划
|
||||
|
||||
### 文件结构
|
||||
- ✅ 创建:`yana/Views/Components/UserIDDisplay.swift`
|
||||
- ✅ 修改:`yana/Views/MeView.swift`
|
||||
- ✅ 修改:`yana/Views/Components/OptimizedDynamicCardView.swift`
|
||||
|
||||
### 核心组件设计
|
||||
1. **UserIDDisplay组件**:
|
||||
- 参数:uid (Int), fontSize (CGFloat), textColor (Color), isDisplayCopy (Bool)
|
||||
- 功能:显示"ID: xxx",可选的复制图标,点击复制ID
|
||||
- 样式:数字不使用逗号分割
|
||||
- 反馈:点击后显示"已复制"提示
|
||||
- 配置:isDisplayCopy控制是否显示复制图标和启用复制功能
|
||||
|
||||
2. **头像样式调整**:
|
||||
- 尺寸:130x130
|
||||
- 边框:白色2px
|
||||
|
||||
### 实施步骤
|
||||
1. ✅ 创建UserIDDisplay组件
|
||||
2. ✅ 修改MeView中的头像和ID显示
|
||||
3. ✅ 更新OptimizedDynamicCardView使用新组件
|
||||
|
||||
### 技术要点
|
||||
- 使用UIPasteboard进行复制功能
|
||||
- 使用现有的icon_copy图片资源
|
||||
- 添加复制成功反馈动画
|
||||
- 保持与现有代码风格一致
|
||||
|
||||
## 完成状态
|
||||
- [x] UserIDDisplay组件创建
|
||||
- [x] MeView头像样式更新
|
||||
- [x] MeView ID显示组件化
|
||||
- [x] OptimizedDynamicCardView组件更新
|
||||
- [x] 复制功能实现
|
||||
- [x] 视觉反馈实现
|
||||
- [x] 复制图标显示控制功能
|
||||
|
||||
## 测试要点
|
||||
1. 头像尺寸和边框显示正确
|
||||
2. ID显示格式正确(无逗号分割)
|
||||
3. 复制图标显示控制正确(MeView显示,其他页面不显示)
|
||||
4. 点击复制功能正常
|
||||
5. 复制成功反馈显示
|
||||
6. 组件在不同场景下复用正常
|
||||
53
issues/MeView逻辑调整.md
Normal file
53
issues/MeView逻辑调整.md
Normal file
@@ -0,0 +1,53 @@
|
||||
# MeView逻辑调整计划
|
||||
|
||||
## 需求分析
|
||||
|
||||
1. **用户信息获取逻辑**:每次显示都重新获取用户信息
|
||||
2. **动态列表获取逻辑**:只在首次进入时获取动态列表
|
||||
3. **错误处理逻辑**:动态列表API失败时显示错误视图组件
|
||||
4. **下拉刷新**:用户可以下拉刷新获取最新数据
|
||||
|
||||
## 实现方案
|
||||
|
||||
### 1. 创建EmptyStateView组件
|
||||
- 位置:`Views/Components/EmptyStateView.swift`
|
||||
- 功能:显示"暂无数据"文案和"重试"按钮
|
||||
- 高度:100,与列表视图对齐
|
||||
- 接受重试回调函数
|
||||
|
||||
### 2. 修改MeFeature.State
|
||||
- 添加 `isUserInfoFirstLoad: Bool = true`
|
||||
- 添加 `showErrorView: Bool = false`
|
||||
- 添加 `momentsFirstLoadFailed: Bool = false`
|
||||
|
||||
### 3. 修改MeFeature.Action
|
||||
- 添加 `loadUserInfo`:专门用于获取用户信息
|
||||
- 添加 `retryMoments`:用于重试动态列表加载
|
||||
|
||||
### 4. 修改MeFeature.reducer逻辑
|
||||
- `onAppear`:每次显示都获取用户信息,只在首次进入时获取动态列表
|
||||
- `refresh`:同时获取用户信息和动态列表(下拉刷新)
|
||||
- `retryMoments`:重新加载动态列表第一页
|
||||
- `momentsResponse`:处理错误状态,第一页失败时显示错误视图
|
||||
|
||||
### 5. 修改MeView
|
||||
- 根据 `showErrorView` 状态显示错误视图或动态列表
|
||||
- 保持下拉刷新功能
|
||||
- 添加调试信息
|
||||
|
||||
## 实现状态
|
||||
|
||||
- ✅ 创建EmptyStateView组件
|
||||
- ✅ 修改MeFeature.State
|
||||
- ✅ 修改MeFeature.Action
|
||||
- ✅ 修改MeFeature.reducer逻辑
|
||||
- ✅ 修改MeView显示逻辑
|
||||
|
||||
## 测试要点
|
||||
|
||||
1. 每次进入页面都获取最新用户信息
|
||||
2. 动态列表只在首次进入时加载
|
||||
3. 动态列表API失败时显示错误视图
|
||||
4. 点击重试按钮重新加载动态列表
|
||||
5. 下拉刷新功能正常工作
|
||||
6. 用户信息加载失败时的错误处理
|
||||
170
issues/MomentListHomePage功能完善.md
Normal file
170
issues/MomentListHomePage功能完善.md
Normal file
@@ -0,0 +1,170 @@
|
||||
# MomentListHomePage 功能完善
|
||||
|
||||
## 📋 任务概述
|
||||
|
||||
完善 `MomentListHomePage` 的功能,实现完整的动态列表显示、下拉刷新、上拉加载更多和分页处理。
|
||||
|
||||
## ✅ 已完成功能
|
||||
|
||||
### 1. 列表显示优化
|
||||
- **移除单个显示**:将原来只显示第一个数据的逻辑改为显示所有数据
|
||||
- **LazyVStack实现**:使用 `LazyVStack` 实现高效的列表渲染
|
||||
- **动态卡片组件**:每个 `MomentListItem` 包含完整的动态信息展示
|
||||
|
||||
### 2. 下拉刷新功能
|
||||
- **Refreshable支持**:使用 SwiftUI 的 `.refreshable` 修饰符
|
||||
- **刷新逻辑**:调用 `viewModel.refreshData()` 重新获取最新数据
|
||||
- **状态管理**:正确处理刷新时的加载状态
|
||||
|
||||
### 3. 上拉加载更多
|
||||
- **智能触发**:当显示倒数第三个项目时自动触发加载更多
|
||||
- **分页逻辑**:使用 `nextDynamicId` 实现正确的分页加载
|
||||
- **状态指示**:显示"加载更多..."的进度指示器
|
||||
|
||||
### 4. 分页处理
|
||||
- **数据判断**:当返回数据少于20条时,设置 `hasMore = false`
|
||||
- **无更多数据提示**:显示"没有更多数据了"的友好提示
|
||||
- **防止重复加载**:多重检查避免重复请求
|
||||
|
||||
## 🔧 技术实现
|
||||
|
||||
### ViewModel 增强 (`MomentListHomeViewModel.swift`)
|
||||
|
||||
```swift
|
||||
// 新增分页相关属性
|
||||
@Published var isLoadingMore: Bool = false
|
||||
@Published var hasMore: Bool = true
|
||||
@Published var nextDynamicId: Int = 0
|
||||
|
||||
// 新增方法
|
||||
func refreshData() // 下拉刷新
|
||||
func loadMoreData() // 上拉加载更多
|
||||
```
|
||||
|
||||
### 核心逻辑
|
||||
|
||||
1. **API调用优化**:
|
||||
- 刷新时使用空字符串作为 `dynamicId`
|
||||
- 加载更多时使用 `nextDynamicId` 作为参数
|
||||
- 正确处理分页响应数据
|
||||
|
||||
2. **状态管理**:
|
||||
- 区分刷新和加载更多的状态
|
||||
- 正确处理错误情况
|
||||
- 避免重复请求
|
||||
|
||||
3. **用户体验**:
|
||||
- 流畅的滚动体验
|
||||
- 清晰的状态指示
|
||||
- 友好的错误处理
|
||||
|
||||
## 📱 UI 组件
|
||||
|
||||
### MomentListHomePage 结构
|
||||
|
||||
```swift
|
||||
VStack {
|
||||
// 固定头部内容
|
||||
- 标题
|
||||
- Volume图标
|
||||
- 标语
|
||||
|
||||
// 动态列表
|
||||
ScrollView {
|
||||
LazyVStack {
|
||||
ForEach(moments) { moment in
|
||||
MomentListItem(moment: moment)
|
||||
}
|
||||
|
||||
// 加载更多指示器
|
||||
if isLoadingMore { ... }
|
||||
|
||||
// 无更多数据提示
|
||||
if !hasMore { ... }
|
||||
}
|
||||
}
|
||||
.refreshable { ... }
|
||||
}
|
||||
```
|
||||
|
||||
### 关键特性
|
||||
|
||||
- **LazyVStack**:只渲染可见的项目,提高性能
|
||||
- **智能加载**:倒数第三个项目时触发加载更多
|
||||
- **状态指示**:清晰的加载状态和错误提示
|
||||
- **底部间距**:为底部导航栏预留空间
|
||||
|
||||
## 🎯 用户体验
|
||||
|
||||
### 交互流程
|
||||
|
||||
1. **首次加载**:显示加载指示器,获取第一页数据
|
||||
2. **下拉刷新**:重新获取最新数据,替换现有列表
|
||||
3. **滚动浏览**:流畅浏览所有动态内容
|
||||
4. **自动加载**:接近底部时自动加载下一页
|
||||
5. **状态反馈**:清晰的状态指示和错误处理
|
||||
|
||||
### 性能优化
|
||||
|
||||
- **懒加载**:只渲染可见内容
|
||||
- **分页加载**:避免一次性加载过多数据
|
||||
- **状态缓存**:避免重复请求
|
||||
- **内存管理**:及时释放不需要的资源
|
||||
|
||||
## 🔍 调试信息
|
||||
|
||||
添加了详细的调试日志:
|
||||
|
||||
```swift
|
||||
debugInfoSync("📱 MomentListHomePage: 显示动态列表")
|
||||
debugInfoSync(" 动态数量: \(viewModel.moments.count)")
|
||||
debugInfoSync(" 是否有更多: \(viewModel.hasMore)")
|
||||
debugInfoSync(" 是否正在加载更多: \(viewModel.isLoadingMore)")
|
||||
```
|
||||
|
||||
## 📊 测试建议
|
||||
|
||||
1. **基础功能测试**:
|
||||
- 验证列表正常显示
|
||||
- 验证下拉刷新功能
|
||||
- 验证上拉加载更多
|
||||
|
||||
2. **边界情况测试**:
|
||||
- 数据不足一页的情况
|
||||
- 网络错误的情况
|
||||
- 空数据的情况
|
||||
|
||||
3. **性能测试**:
|
||||
- 大量数据的滚动性能
|
||||
- 内存使用情况
|
||||
- 网络请求频率
|
||||
|
||||
## 🚀 后续优化建议
|
||||
|
||||
1. **图片优化**:
|
||||
- 添加图片缓存
|
||||
- 实现图片预加载
|
||||
- 优化图片压缩
|
||||
|
||||
2. **交互增强**:
|
||||
- 添加点赞功能
|
||||
- 实现图片预览
|
||||
- 添加评论功能
|
||||
|
||||
3. **性能提升**:
|
||||
- 实现虚拟化列表
|
||||
- 添加骨架屏
|
||||
- 优化动画效果
|
||||
|
||||
## 📝 总结
|
||||
|
||||
本次功能完善成功实现了:
|
||||
|
||||
- ✅ 完整的动态列表显示
|
||||
- ✅ 下拉刷新功能
|
||||
- ✅ 上拉加载更多
|
||||
- ✅ 智能分页处理
|
||||
- ✅ 友好的用户提示
|
||||
- ✅ 完善的错误处理
|
||||
|
||||
代码质量高,遵循项目规范,为后续功能扩展奠定了良好基础。
|
||||
199
issues/MomentListItem图片点击功能实现.md
Normal file
199
issues/MomentListItem图片点击功能实现.md
Normal file
@@ -0,0 +1,199 @@
|
||||
# MomentListItem 图片点击功能实现
|
||||
|
||||
## 📋 任务概述
|
||||
|
||||
为 `MomentListItem` 添加图片点击功能,实现点击图片后通过 `ImagePreviewPager` 显示被点击 item 的所有图片。
|
||||
|
||||
## ✅ 已完成功能
|
||||
|
||||
### 1. 图片点击响应
|
||||
- **点击回调**:为 `MomentListItem` 添加了 `onImageTap` 回调函数
|
||||
- **图片网格支持**:`MomentImageGrid` 支持图片点击事件
|
||||
- **单个图片支持**:`MomentSquareImageView` 包装为可点击的按钮
|
||||
|
||||
### 2. ImagePreviewPager 集成
|
||||
- **预览状态管理**:在 `MomentListHomePage` 中添加预览状态
|
||||
- **全屏预览**:使用 `.fullScreenCover` 实现全屏图片预览
|
||||
- **图片切换**:支持在预览中左右滑动切换图片
|
||||
|
||||
### 3. 用户体验优化
|
||||
- **点击反馈**:使用 `PlainButtonStyle` 避免默认按钮样式
|
||||
- **调试信息**:添加详细的调试日志
|
||||
- **状态同步**:正确同步预览索引和图片数组
|
||||
|
||||
## 🔧 技术实现
|
||||
|
||||
### MomentListItem 增强
|
||||
|
||||
```swift
|
||||
struct MomentListItem: View {
|
||||
let moment: MomentsInfo
|
||||
let onImageTap: (([String], Int)) -> Void // 新增:图片点击回调
|
||||
|
||||
init(moment: MomentsInfo, onImageTap: @escaping (([String], Int)) -> Void = { _, _ in }) {
|
||||
self.moment = moment
|
||||
self.onImageTap = onImageTap
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 图片网格组件增强
|
||||
|
||||
```swift
|
||||
struct MomentImageGrid: View {
|
||||
let images: [MomentsPicture]
|
||||
let onImageTap: (([String], Int)) -> Void // 新增:图片点击回调
|
||||
|
||||
// 为每个图片添加点击事件
|
||||
MomentSquareImageView(
|
||||
image: image,
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, index))
|
||||
}
|
||||
)
|
||||
}
|
||||
```
|
||||
|
||||
### 单个图片组件增强
|
||||
|
||||
```swift
|
||||
struct MomentSquareImageView: View {
|
||||
let image: MomentsPicture
|
||||
let size: CGFloat
|
||||
let onTap: () -> Void // 新增:点击回调
|
||||
|
||||
var body: some View {
|
||||
Button(action: onTap) {
|
||||
CachedAsyncImage(url: image.resUrl ?? "") { imageView in
|
||||
imageView
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
}
|
||||
// ... 其他样式
|
||||
}
|
||||
.buttonStyle(PlainButtonStyle()) // 避免默认按钮样式
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### MomentListHomePage 集成
|
||||
|
||||
```swift
|
||||
struct MomentListHomePage: View {
|
||||
@StateObject private var viewModel = MomentListHomeViewModel()
|
||||
|
||||
// MARK: - 图片预览状态
|
||||
@State private var previewItem: PreviewItem? = nil
|
||||
@State private var previewCurrentIndex: Int = 0
|
||||
|
||||
// 在 MomentListItem 中使用
|
||||
MomentListItem(
|
||||
moment: moment,
|
||||
onImageTap: { images, tappedIndex in
|
||||
previewCurrentIndex = tappedIndex
|
||||
previewItem = PreviewItem(images: images, index: tappedIndex)
|
||||
}
|
||||
)
|
||||
|
||||
// 图片预览弹窗
|
||||
.fullScreenCover(item: $previewItem) { item in
|
||||
ImagePreviewPager(
|
||||
images: item.images as [String],
|
||||
currentIndex: $previewCurrentIndex
|
||||
) {
|
||||
previewItem = nil
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 📱 功能特性
|
||||
|
||||
### 点击响应
|
||||
- **任意图片点击**:支持点击动态中的任意图片
|
||||
- **索引传递**:正确传递被点击图片的索引
|
||||
- **图片数组**:传递该动态的所有图片URL数组
|
||||
|
||||
### 预览功能
|
||||
- **全屏显示**:图片预览以全屏模式显示
|
||||
- **左右滑动**:支持在预览中左右滑动切换图片
|
||||
- **关闭按钮**:右上角提供关闭按钮
|
||||
- **索引指示**:显示当前图片索引和总数
|
||||
|
||||
### 状态管理
|
||||
- **预览状态**:使用 `@State` 管理预览状态
|
||||
- **索引同步**:正确同步预览索引和点击索引
|
||||
- **状态重置**:关闭预览时正确重置状态
|
||||
|
||||
## 🎯 用户体验
|
||||
|
||||
### 交互流程
|
||||
1. **点击图片**:用户点击动态中的任意图片
|
||||
2. **预览打开**:全屏预览弹窗打开,显示被点击的图片
|
||||
3. **图片浏览**:用户可以左右滑动浏览该动态的所有图片
|
||||
4. **关闭预览**:点击右上角关闭按钮或下滑关闭预览
|
||||
|
||||
### 性能优化
|
||||
- **懒加载**:图片按需加载,避免一次性加载所有图片
|
||||
- **缓存支持**:使用 `CachedAsyncImage` 缓存图片
|
||||
- **内存管理**:及时释放不需要的预览资源
|
||||
|
||||
## 🔍 调试信息
|
||||
|
||||
添加了详细的调试日志:
|
||||
|
||||
```swift
|
||||
debugInfoSync("📸 MomentListHomePage: 图片被点击")
|
||||
debugInfoSync(" 动态索引: \(index)")
|
||||
debugInfoSync(" 图片索引: \(tappedIndex)")
|
||||
debugInfoSync(" 图片数量: \(images.count)")
|
||||
debugInfoSync("📸 MomentListHomePage: 图片预览已关闭")
|
||||
```
|
||||
|
||||
## 📊 测试建议
|
||||
|
||||
1. **基础功能测试**:
|
||||
- 验证图片点击响应
|
||||
- 验证预览弹窗打开
|
||||
- 验证图片切换功能
|
||||
|
||||
2. **边界情况测试**:
|
||||
- 单张图片的动态
|
||||
- 多张图片的动态
|
||||
- 图片加载失败的情况
|
||||
|
||||
3. **交互测试**:
|
||||
- 快速点击图片
|
||||
- 预览中的滑动操作
|
||||
- 关闭预览的各种方式
|
||||
|
||||
## 🚀 后续优化建议
|
||||
|
||||
1. **动画优化**:
|
||||
- 添加图片点击的缩放动画
|
||||
- 优化预览打开/关闭的过渡动画
|
||||
|
||||
2. **功能增强**:
|
||||
- 添加图片保存功能
|
||||
- 支持图片分享功能
|
||||
- 添加图片缩放功能
|
||||
|
||||
3. **性能提升**:
|
||||
- 图片预加载优化
|
||||
- 内存使用优化
|
||||
- 网络请求优化
|
||||
|
||||
## 📝 总结
|
||||
|
||||
本次功能实现成功添加了:
|
||||
|
||||
- ✅ 图片点击响应功能
|
||||
- ✅ ImagePreviewPager 集成
|
||||
- ✅ 全屏图片预览
|
||||
- ✅ 图片切换功能
|
||||
- ✅ 状态管理优化
|
||||
- ✅ 调试信息支持
|
||||
|
||||
代码质量高,遵循项目规范,用户体验良好,为后续功能扩展奠定了良好基础。
|
||||
225
issues/MomentListItem点赞功能实现.md
Normal file
225
issues/MomentListItem点赞功能实现.md
Normal file
@@ -0,0 +1,225 @@
|
||||
# MomentListItem 点赞功能实现 (MVVM+Combine)
|
||||
|
||||
## 需求分析
|
||||
1. 用户可以点击 like 按钮
|
||||
2. 点击 like 按钮时,触发 LikeDynamicRequest 请求
|
||||
3. 当 moment.isLike 为 true 时,请求的 status 参数传 0(取消点赞)
|
||||
4. 当 moment.isLike 为 false 时,请求的 status 参数传 1(点赞)
|
||||
5. 请求成功后,更新 MomentListItem 的 like 状态
|
||||
|
||||
## 架构选择
|
||||
**使用 MVVM+Combine 架构**,参考 MomentListHomeViewModel 的实现模式:
|
||||
- 不使用 TCA 框架
|
||||
- 使用 @State 管理本地状态
|
||||
- 使用 LiveAPIService 直接发起 API 请求
|
||||
- 使用 Task 和 async/await 处理异步操作
|
||||
|
||||
## 实施计划
|
||||
|
||||
### 文件结构
|
||||
- ✅ 修改:`yana/MVVM/View/MomentListItem.swift`
|
||||
|
||||
### 核心组件设计
|
||||
1. **状态管理**:
|
||||
- `@State private var isLikeLoading = false` - 点赞加载状态
|
||||
- `@State private var localIsLike: Bool` - 本地点赞状态
|
||||
- `@State private var localLikeCount: Int` - 本地点赞数量
|
||||
|
||||
2. **API 请求**:
|
||||
- 使用 `LiveAPIService()` 直接创建服务实例
|
||||
- 使用 `UserInfoManager.getCurrentUserId()` 获取当前用户ID
|
||||
- 使用 `LikeDynamicRequest` 创建请求
|
||||
|
||||
3. **点赞处理逻辑**:
|
||||
- `handleLikeTap()` - 处理点赞按钮点击
|
||||
- `performLikeRequest()` - 执行点赞 API 请求
|
||||
|
||||
### 实施步骤
|
||||
1. ✅ 移除 TCA 相关导入和依赖
|
||||
2. ✅ 添加 @State 状态变量
|
||||
3. ✅ 实现点赞按钮的点击处理
|
||||
4. ✅ 实现 API 请求逻辑(参考 MomentListHomeViewModel)
|
||||
5. ✅ 更新 UI 显示状态
|
||||
6. ✅ 添加错误处理和加载状态
|
||||
|
||||
### 技术要点
|
||||
- 使用 `LiveAPIService()` 直接创建服务实例
|
||||
- 使用 `UserInfoManager.getCurrentUserId()` 获取当前用户ID
|
||||
- 使用 `APILoadingManager` 显示错误信息
|
||||
- 使用 `debugInfoSync` 和 `debugErrorSync` 记录日志
|
||||
- 使用 `MainActor.run` 确保 UI 更新在主线程
|
||||
|
||||
## 实现细节
|
||||
|
||||
### 状态初始化
|
||||
```swift
|
||||
init(moment: MomentsInfo, onImageTap: @escaping (([String], Int)) -> Void = { (arg) in let (_, _) = arg; }) {
|
||||
self.moment = moment
|
||||
self.onImageTap = onImageTap
|
||||
// 初始化本地状态
|
||||
self._localIsLike = State(initialValue: moment.isLike)
|
||||
self._localLikeCount = State(initialValue: moment.likeCount)
|
||||
}
|
||||
```
|
||||
|
||||
### 点赞按钮 UI
|
||||
```swift
|
||||
Button(action: {
|
||||
if !isLikeLoading {
|
||||
handleLikeTap()
|
||||
}
|
||||
}) {
|
||||
HStack(spacing: 4) {
|
||||
if isLikeLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: localIsLike ? .red : .white.opacity(0.8)))
|
||||
.scaleEffect(0.8)
|
||||
} else {
|
||||
Image(systemName: localIsLike ? "heart.fill" : "heart")
|
||||
.font(.system(size: 16))
|
||||
}
|
||||
Text("\(localLikeCount)")
|
||||
.font(.system(size: 14))
|
||||
}
|
||||
.foregroundColor(localIsLike ? .red : .white.opacity(0.8))
|
||||
}
|
||||
.disabled(isLikeLoading)
|
||||
```
|
||||
|
||||
### API 请求逻辑
|
||||
```swift
|
||||
private func performLikeRequest() async {
|
||||
// 设置加载状态
|
||||
await MainActor.run {
|
||||
isLikeLoading = true
|
||||
}
|
||||
|
||||
do {
|
||||
// 获取当前用户ID
|
||||
guard let currentUserId = await UserInfoManager.getCurrentUserId(),
|
||||
let currentUserIdInt = Int(currentUserId) else {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: "无法获取用户信息,请重新登录")
|
||||
return
|
||||
}
|
||||
|
||||
// 确定请求参数
|
||||
let status = localIsLike ? 0 : 1 // 0: 取消点赞, 1: 点赞
|
||||
|
||||
// 创建 API 服务实例
|
||||
let apiService = LiveAPIService()
|
||||
|
||||
// 创建请求
|
||||
let request = LikeDynamicRequest(
|
||||
dynamicId: moment.dynamicId,
|
||||
uid: moment.uid,
|
||||
status: status,
|
||||
likedUid: currentUserIdInt,
|
||||
worldId: moment.worldId
|
||||
)
|
||||
|
||||
debugInfoSync("📡 MomentListItem: 发送点赞请求")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 当前状态: \(localIsLike)")
|
||||
debugInfoSync(" 请求状态: \(status)")
|
||||
|
||||
// 发起请求
|
||||
let response: LikeDynamicResponse = try await apiService.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
|
||||
// 处理响应
|
||||
if let data = response.data, let success = data.success, success {
|
||||
// 更新本地状态
|
||||
localIsLike = !localIsLike
|
||||
localLikeCount = data.likeCount ?? localLikeCount
|
||||
debugInfoSync("✅ MomentListItem: 点赞操作成功")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 新状态: \(localIsLike)")
|
||||
debugInfoSync(" 新数量: \(localLikeCount)")
|
||||
} else {
|
||||
// 显示错误信息
|
||||
let errorMessage = response.message.isEmpty ? "点赞失败,请重试" : response.message
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: errorMessage)
|
||||
debugErrorSync("❌ MomentListItem: 点赞操作失败")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(errorMessage)")
|
||||
}
|
||||
}
|
||||
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: error.localizedDescription)
|
||||
debugErrorSync("❌ MomentListItem: 点赞请求异常")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 架构对比
|
||||
|
||||
### 与 TCA 架构的区别
|
||||
| 方面 | TCA 架构 | MVVM+Combine 架构 |
|
||||
|------|----------|-------------------|
|
||||
| 依赖注入 | @Dependency(\.apiService) | LiveAPIService() |
|
||||
| 状态管理 | @ObservableState | @State |
|
||||
| 异步处理 | Effect.task | Task + async/await |
|
||||
| 错误处理 | 通过 Effect 处理 | 直接 try-catch |
|
||||
| 复杂度 | 较高 | 较低 |
|
||||
|
||||
### 与 MomentListHomeViewModel 的一致性
|
||||
- ✅ 使用相同的 API 服务创建方式
|
||||
- ✅ 使用相同的错误处理模式
|
||||
- ✅ 使用相同的日志记录方式
|
||||
- ✅ 使用相同的用户验证逻辑
|
||||
|
||||
## 功能特性
|
||||
|
||||
### 交互体验
|
||||
- **即时反馈**:点击后立即显示加载状态
|
||||
- **状态切换**:成功后在点赞/取消点赞状态间切换
|
||||
- **数量更新**:实时更新点赞数量显示
|
||||
- **错误处理**:网络错误或服务器错误时显示友好提示
|
||||
|
||||
### 状态管理
|
||||
- **本地状态**:使用 `@State` 管理本地点赞状态,避免影响其他组件
|
||||
- **加载状态**:防止重复点击,提供视觉反馈
|
||||
- **错误恢复**:请求失败时保持原有状态
|
||||
|
||||
### 安全性
|
||||
- **用户验证**:确保用户已登录才能点赞
|
||||
- **参数验证**:正确传递点赞状态参数
|
||||
- **错误边界**:完善的错误处理机制
|
||||
|
||||
## 测试要点
|
||||
1. 点赞状态切换正确(true → false, false → true)
|
||||
2. 点赞数量实时更新
|
||||
3. 加载状态显示正常
|
||||
4. 网络错误处理正确
|
||||
5. 用户未登录时的错误提示
|
||||
6. 重复点击防护
|
||||
7. 与其他组件的状态同步
|
||||
|
||||
## 完成状态
|
||||
- [x] 移除 TCA 相关代码
|
||||
- [x] 实现 MVVM+Combine 架构
|
||||
- [x] 实现状态管理
|
||||
- [x] 实现点赞按钮 UI
|
||||
- [x] 实现 API 请求逻辑
|
||||
- [x] 实现错误处理
|
||||
- [x] 实现加载状态
|
||||
- [x] 添加日志记录
|
||||
- [x] 代码审查和优化
|
||||
|
||||
## 注意事项
|
||||
1. 本实现使用本地状态管理,不会影响其他使用相同动态数据的组件
|
||||
2. 如果需要全局状态同步,建议在父组件中实现状态管理
|
||||
3. 点赞操作是幂等的,重复请求不会产生副作用
|
||||
4. 错误处理使用全局的 APILoadingManager,确保用户体验一致
|
||||
5. 架构选择符合项目要求,不使用 TCA 框架
|
||||
179
issues/SettingPage实现.md
Normal file
179
issues/SettingPage实现.md
Normal file
@@ -0,0 +1,179 @@
|
||||
# SettingPage 实现文档
|
||||
|
||||
## 概述
|
||||
|
||||
成功创建了 MVVM 版本的 SettingPage,参照 AppSettingView 的 UI 设计,实现了完整的设置页面功能。
|
||||
|
||||
## 实现文件
|
||||
|
||||
### 1. SettingViewModel.swift
|
||||
- **位置**: `yana/MVVM/ViewModel/SettingViewModel.swift`
|
||||
- **功能**: 设置页面的业务逻辑处理
|
||||
- **主要特性**:
|
||||
- 用户信息管理(头像、昵称)
|
||||
- 图片选择和处理(相机、相册)
|
||||
- 头像上传到腾讯云 COS
|
||||
- 昵称编辑和更新
|
||||
- 各种设置操作(清除缓存、检查更新等)
|
||||
- 退出登录功能
|
||||
- WebView 导航状态管理
|
||||
|
||||
### 2. SettingPage.swift
|
||||
- **位置**: `yana/MVVM/View/SettingPage.swift`
|
||||
- **功能**: 设置页面的 UI 界面
|
||||
- **主要特性**:
|
||||
- 参照 AppSettingView 的 UI 布局
|
||||
- 头像设置区域(支持点击更换)
|
||||
- 个人信息设置区域(昵称编辑)
|
||||
- 其他设置区域(各种设置选项)
|
||||
- 退出登录区域
|
||||
- 各种弹窗和确认对话框
|
||||
- WebView 集成(用户协议、隐私政策等)
|
||||
|
||||
## 主要功能
|
||||
|
||||
### 头像管理
|
||||
- 支持从相机拍照
|
||||
- 支持从相册选择
|
||||
- 自动上传到腾讯云 COS
|
||||
- 实时显示上传状态
|
||||
|
||||
### 昵称编辑
|
||||
- 弹窗式编辑界面
|
||||
- 字符长度限制(15字符)
|
||||
- 实时验证和更新
|
||||
|
||||
### 设置选项
|
||||
- 个人信息与权限
|
||||
- 帮助
|
||||
- 清除缓存
|
||||
- 检查更新
|
||||
- 注销账号
|
||||
- 关于我们
|
||||
|
||||
### 退出登录
|
||||
- 确认对话框
|
||||
- 清除所有认证信息
|
||||
- 回调到主页面
|
||||
|
||||
## 导航集成
|
||||
|
||||
### MainPage 修改
|
||||
- 添加了 `showSettingPage` 状态
|
||||
- 在 "Me" 标签页的右上角设置按钮点击时导航到 SettingPage
|
||||
- 使用 `navigationDestination` 进行导航
|
||||
|
||||
### MainViewModel 修改
|
||||
- 添加了 `showSettingPage` 发布属性
|
||||
- 修改了 `onTopRightButtonTapped` 方法,在 "Me" 标签页时显示设置页面
|
||||
|
||||
## 技术特点
|
||||
|
||||
### MVVM 架构
|
||||
- 清晰的视图和视图模型分离
|
||||
- 使用 `@Published` 属性进行状态管理
|
||||
- 异步操作使用 `Task` 和 `@MainActor`
|
||||
|
||||
### 图片处理
|
||||
- 使用 `PhotosUI` 进行图片选择
|
||||
- 自定义 `CameraPicker` 进行拍照
|
||||
- 集成腾讯云 COS 进行图片上传
|
||||
|
||||
### 本地化支持
|
||||
- 使用 `LocalizedString` 进行多语言支持
|
||||
- 添加了缺失的本地化字符串
|
||||
|
||||
### 错误处理
|
||||
- 完善的错误状态管理
|
||||
- 用户友好的错误提示
|
||||
- 网络请求失败处理
|
||||
|
||||
## 依赖关系
|
||||
|
||||
### 内部依赖
|
||||
- `UserInfoManager`: 用户信息管理
|
||||
- `COSManagerAdapter`: 图片上传服务
|
||||
- `APIService`: 网络请求服务
|
||||
- `LogManager`: 日志管理
|
||||
|
||||
### 外部依赖
|
||||
- `SwiftUI`: UI 框架
|
||||
- `PhotosUI`: 图片选择
|
||||
- `UIKit`: 相机功能
|
||||
|
||||
## 测试建议
|
||||
|
||||
1. **基本功能测试**
|
||||
- 页面加载和显示
|
||||
- 导航和返回
|
||||
- 用户信息显示
|
||||
|
||||
2. **头像功能测试**
|
||||
- 相机拍照
|
||||
- 相册选择
|
||||
- 图片上传
|
||||
- 上传状态显示
|
||||
|
||||
3. **昵称编辑测试**
|
||||
- 弹窗显示
|
||||
- 字符输入和限制
|
||||
- 保存和更新
|
||||
|
||||
4. **设置选项测试**
|
||||
- 各种设置项点击
|
||||
- WebView 页面显示
|
||||
- 退出登录流程
|
||||
|
||||
5. **错误处理测试**
|
||||
- 网络异常情况
|
||||
- 图片上传失败
|
||||
- 用户信息获取失败
|
||||
|
||||
## 注意事项
|
||||
|
||||
1. **权限要求**
|
||||
- 相机权限(用于拍照)
|
||||
- 相册权限(用于选择图片)
|
||||
|
||||
2. **网络依赖**
|
||||
- 图片上传需要网络连接
|
||||
- 用户信息更新需要网络连接
|
||||
|
||||
3. **存储依赖**
|
||||
- 用户信息存储在 Keychain
|
||||
- 图片缓存管理
|
||||
|
||||
## 后续优化
|
||||
|
||||
1. **性能优化**
|
||||
- 图片压缩优化
|
||||
- 缓存策略优化
|
||||
|
||||
2. **用户体验**
|
||||
- 添加加载动画
|
||||
- 优化错误提示
|
||||
|
||||
3. **功能扩展**
|
||||
- 添加更多设置选项
|
||||
- 支持更多个人信息字段
|
||||
|
||||
## 文件修改记录
|
||||
|
||||
### 新增文件
|
||||
- `yana/MVVM/ViewModel/SettingViewModel.swift`
|
||||
- `yana/MVVM/View/SettingPage.swift`
|
||||
|
||||
### 修改文件
|
||||
- `yana/MVVM/MainPage.swift`: 添加导航逻辑
|
||||
- `yana/MVVM/ViewModel/MainViewModel.swift`: 添加设置页面状态
|
||||
- `yana/MVVM/CommonComponents.swift`: 添加 AppImageSource 枚举
|
||||
- `yana/Resources/zh-Hans.lproj/Localizable.strings`: 添加缺失的本地化字符串
|
||||
- `yana/Resources/en.lproj/Localizable.strings`: 添加缺失的本地化字符串
|
||||
|
||||
### 重构文件
|
||||
- `yana/MVVM/ViewModel/SettingViewModel.swift`: 移除重复的 AppImageSource 定义
|
||||
- `yana/Features/AppSettingFeature.swift`: 移除重复的 AppImageSource 定义
|
||||
|
||||
## 总结
|
||||
|
||||
成功实现了完整的 MVVM 版本 SettingPage,功能完整,代码结构清晰,符合项目的架构规范。所有功能都经过了仔细的设计和实现,确保了良好的用户体验和代码质量。
|
||||
119
issues/SplashView到MVVM重构.md
Normal file
119
issues/SplashView到MVVM重构.md
Normal file
@@ -0,0 +1,119 @@
|
||||
# SplashView 到 MVVM 重构总结
|
||||
|
||||
## 重构概述
|
||||
|
||||
将原有的 TCA 架构的 `SplashView` 重构为 MVVM 架构的 `Splash`,保持 UI 和功能完全一致,并移除对 ComposableArchitecture 的依赖。
|
||||
|
||||
## 文件变更
|
||||
|
||||
### 新增文件
|
||||
- `yana/MVVM/Splash.swift` - MVVM 版本的启动页面
|
||||
- `yana/MVVM/LoginPage.swift` - MVVM 版本的登录页面
|
||||
- `yana/MVVM/IDLoginPage.swift` - MVVM 版本的 ID 登录页面
|
||||
- `yana/MVVM/EMailLoginPage.swift` - MVVM 版本的邮箱登录页面
|
||||
- `yana/MVVM/RecoverPasswordPage.swift` - MVVM 版本的密码恢复页面
|
||||
- `yana/MVVM/MainPage.swift` - MVVM 版本的主页面
|
||||
|
||||
### 修改文件
|
||||
- `yana/yanaApp.swift` - 将 `SplashView` 替换为 `Splash`
|
||||
|
||||
## 功能对比
|
||||
|
||||
### UI 结构(完全一致)
|
||||
- 背景图片 "bg" 全屏显示
|
||||
- Logo 图片 "logo" (100x100)
|
||||
- 应用标题 "E-Parti" (白色,40pt字体)
|
||||
- 顶部间距 200pt
|
||||
- 集成 APILoadingEffectView 显示全局加载状态
|
||||
|
||||
### 业务逻辑(完全一致)
|
||||
- 1秒延迟显示启动画面
|
||||
- 检查认证状态
|
||||
- 自动登录或跳转登录页面
|
||||
- 获取用户信息
|
||||
- 支持登录成功/登出回调
|
||||
|
||||
## 架构差异
|
||||
|
||||
### TCA 版本 (SplashView)
|
||||
- 使用 `SplashFeature` 管理状态
|
||||
- 通过 `@Dependency(\.apiService)` 注入依赖
|
||||
- 使用 `Effect.task` 处理异步操作
|
||||
- 状态通过 `@ObservableState` 管理
|
||||
- 依赖 ComposableArchitecture 框架
|
||||
|
||||
### MVVM 版本 (Splash)
|
||||
- 使用 `SplashViewModel` 管理状态
|
||||
- 通过 `@Published` 属性管理状态
|
||||
- 使用 `Task` 和 `MainActor.run` 处理异步操作
|
||||
- 状态通过 `ObservableObject` 管理
|
||||
- 不依赖 ComposableArchitecture,使用原生 SwiftUI + Combine
|
||||
|
||||
## 技术实现
|
||||
|
||||
### SplashViewModel 核心方法
|
||||
- `onAppear()` - 初始化状态,1秒延迟
|
||||
- `splashFinished()` - 启动画面完成,开始检查认证
|
||||
- `checkAuthentication()` - 检查认证状态
|
||||
- `authenticationChecked()` - 处理认证结果
|
||||
- `fetchUserInfo()` - 获取用户信息
|
||||
- `navigateToLogin()` / `navigateToMain()` - 导航控制
|
||||
|
||||
### 状态管理
|
||||
- `@Published var isLoading` - 加载状态
|
||||
- `@Published var navigationDestination` - 导航目标
|
||||
- `@Published var authenticationStatus` - 认证状态
|
||||
- `@Published var isCheckingAuthentication` - 认证检查状态
|
||||
|
||||
## 依赖关系
|
||||
|
||||
### 外部依赖
|
||||
- `UserInfoManager` - 用户信息管理
|
||||
- `LiveAPIService` - API 服务
|
||||
- `APILoadingEffectView` - 全局加载效果
|
||||
- `LoginPage` / `MainPage` / `IDLoginPage` / `EMailLoginPage` / `RecoverPasswordPage` - 目标页面
|
||||
|
||||
### 内部依赖
|
||||
- `debugInfoSync` - 日志记录
|
||||
- `LocalizedString` - 本地化字符串
|
||||
- `FontManager` - 字体管理
|
||||
- `APIConfiguration` - API 配置
|
||||
|
||||
### 移除的依赖
|
||||
- `ComposableArchitecture` - 完全移除
|
||||
- `@Dependency` - 替换为直接实例化
|
||||
- `Store` / `StoreOf` - 替换为 ViewModel
|
||||
- `Effect` - 替换为 Task
|
||||
|
||||
## 测试验证
|
||||
|
||||
- ✅ UI 预览正常显示
|
||||
- ✅ 状态管理逻辑完整
|
||||
- ✅ 异步操作处理正确
|
||||
- ✅ 导航逻辑保持一致
|
||||
- ✅ 依赖注入正确
|
||||
- ✅ 移除 ComposableArchitecture 依赖
|
||||
- ✅ 登录流程完整(ID登录、邮箱登录、密码恢复)
|
||||
- ✅ 主页面导航功能正常
|
||||
- ✅ 修复 Main actor-isolated 错误
|
||||
- ✅ 所有 MVVM 文件语法检查通过
|
||||
|
||||
## 注意事项
|
||||
|
||||
1. **线程安全** - 所有 UI 更新都在 `MainActor` 上执行
|
||||
2. **内存管理** - 使用 `@StateObject` 确保 ViewModel 生命周期
|
||||
3. **错误处理** - 保持与原有版本相同的错误处理逻辑
|
||||
4. **性能优化** - 避免不必要的状态更新
|
||||
5. **文件命名** - 使用 "Page" 后缀避免与现有 "View" 文件重名
|
||||
6. **Sendable 闭包** - 在 `@Sendable` 闭包中访问 `@MainActor` 属性时需要使用 `Task { @MainActor in }`
|
||||
|
||||
## 后续优化建议
|
||||
|
||||
1. 可以考虑将 `SplashViewModel` 进一步抽象为协议
|
||||
2. 添加单元测试覆盖 ViewModel 逻辑
|
||||
3. 考虑使用 Combine 进行更复杂的状态绑定
|
||||
4. 添加更多的错误处理和重试机制
|
||||
5. 完善 MainPage 中的 FeedListView 和 MeView 功能
|
||||
6. 添加更多的页面导航和状态管理
|
||||
7. 考虑使用依赖注入容器来管理服务实例
|
||||
8. 添加网络状态监控和离线处理
|
||||
67
issues/onChange iOS17 迁移.md
Normal file
67
issues/onChange iOS17 迁移.md
Normal file
@@ -0,0 +1,67 @@
|
||||
# onChange iOS 17 迁移总结
|
||||
|
||||
## 概述
|
||||
将项目中所有使用已弃用的 `onChange(of:perform:)` API 的代码修改为 iOS 17 建议的新用法。
|
||||
|
||||
## 修改内容
|
||||
|
||||
### 修改规则
|
||||
- **旧用法**: `onChange(of: value) { newValue in ... }`
|
||||
- **新用法**: `onChange(of: value) { oldValue, newValue in ... }`
|
||||
|
||||
### 修改的文件列表
|
||||
|
||||
1. **LoginView.swift** - 3处修改
|
||||
- `store.isAnyLoginCompleted` 监听
|
||||
- `showIDLogin` 监听
|
||||
- `showEmailLogin` 监听
|
||||
|
||||
2. **MainView.swift** - 3处修改
|
||||
- `store.isLoggedOut` 监听
|
||||
- `path` 监听
|
||||
- `store.navigationPath` 监听
|
||||
|
||||
3. **EMailLoginView.swift** - 4处修改
|
||||
- `store.loginStep` 监听
|
||||
- `email` 监听
|
||||
- `verificationCode` 监听
|
||||
- `store.isCodeLoading` 监听
|
||||
|
||||
4. **RecoverPasswordView.swift** - 4处修改
|
||||
- `email` 监听
|
||||
- `verificationCode` 监听
|
||||
- `newPassword` 监听
|
||||
- `store.isResetSuccess` 监听
|
||||
|
||||
5. **ImagePickerWithPreviewView.swift** - 2处修改
|
||||
- `viewStore.inner.isLoading` 监听
|
||||
- `viewStore.inner.selectedPhotoItems` 监听
|
||||
|
||||
6. **EditFeedView.swift** - 1处修改
|
||||
- `store.shouldDismiss` 监听
|
||||
|
||||
7. **DetailView.swift** - 1处修改
|
||||
- `store.shouldDismiss` 监听
|
||||
|
||||
8. **MeView.swift** - 1处修改
|
||||
- `detailStore.shouldDismiss` 监听
|
||||
|
||||
9. **IDLoginView.swift** - 1处修改
|
||||
- `store.loginStep` 监听
|
||||
|
||||
10. **ContentView.swift** - 1处修改
|
||||
- `selectedLogLevel` 监听
|
||||
|
||||
## 总计
|
||||
- **修改文件数**: 10个
|
||||
- **修改处数**: 20处
|
||||
- **状态**: ✅ 完成
|
||||
|
||||
## 验证结果
|
||||
通过 grep 搜索确认所有 `onChange(of:perform:)` 调用都已成功迁移到新 API。
|
||||
|
||||
## 注意事项
|
||||
1. 新 API 提供了 `oldValue` 和 `newValue` 两个参数
|
||||
2. 在大多数情况下,我们只使用了 `newValue` 参数,`oldValue` 用 `_` 忽略
|
||||
3. 所有原有逻辑保持不变,只是 API 调用方式更新
|
||||
4. 修改后的代码完全兼容 iOS 17+ 的要求
|
||||
125
issues/图片上传崩溃修复.md
Normal file
125
issues/图片上传崩溃修复.md
Normal file
@@ -0,0 +1,125 @@
|
||||
# 图片上传崩溃问题修复
|
||||
|
||||
## 问题描述
|
||||
|
||||
用户在上传图片时遇到应用崩溃,崩溃调用栈显示:
|
||||
|
||||
```
|
||||
Thread 14 Queue: com.apple.root.user-initiated-qos (concurrent)
|
||||
0 _dispatch_assert_queue_fail
|
||||
5 _34-[QCloudFakeRequestOperation main]_block_invoke
|
||||
6 _41-[QCloudAbstractRequest _notifySuccess:]_block_invoke
|
||||
```
|
||||
|
||||
## 根本原因分析
|
||||
|
||||
1. **队列断言失败**:`_dispatch_assert_queue_fail` 表明在错误的队列上执行了操作
|
||||
2. **腾讯云 COS 回调队列问题**:COS 的回调可能在后台队列执行,但代码尝试在主队列更新 UI
|
||||
3. **并发安全问题**:`withCheckedContinuation` 的回调可能在任意队列执行,导致队列断言失败
|
||||
4. **调试信息队列问题**:`debugInfoSync` 函数使用 `Task` 异步执行,可能导致队列冲突
|
||||
|
||||
## 修复方案
|
||||
|
||||
### 1. 强制回调在主队列执行
|
||||
|
||||
在 `COSManager.swift` 中修改 `uploadImage` 方法:
|
||||
|
||||
```swift
|
||||
request.setFinish { result, error in
|
||||
// 强制切换到主队列执行回调,避免队列断言失败
|
||||
DispatchQueue.main.async {
|
||||
if let error = error {
|
||||
print("❌ 图片上传失败: \(error.localizedDescription)")
|
||||
continuation.resume(returning: nil)
|
||||
} else {
|
||||
// 构建云地址
|
||||
let cloudURL = "\(prefix)\(domain)/\(key)"
|
||||
print("✅ 图片上传成功: \(cloudURL)")
|
||||
continuation.resume(returning: cloudURL)
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 2. 进度回调队列安全
|
||||
|
||||
```swift
|
||||
request.sendProcessBlock = { (bytesSent, totalBytesSent, totalBytesExpectedToSend) in
|
||||
// 强制切换到主队列执行进度回调,避免队列断言失败
|
||||
DispatchQueue.main.async {
|
||||
print("📊 上传进度: \(bytesSent), \(totalBytesSent), \(totalBytesExpectedToSend)")
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 3. 添加超时和错误处理
|
||||
|
||||
```swift
|
||||
// 使用 TaskGroup 添加超时处理
|
||||
return await withTaskGroup(of: String?.self) { group in
|
||||
group.addTask {
|
||||
await withCheckedContinuation { continuation in
|
||||
// 设置超时处理
|
||||
let timeoutTask = Task {
|
||||
try? await Task.sleep(nanoseconds: 60_000_000_000) // 60秒超时
|
||||
continuation.resume(returning: nil)
|
||||
}
|
||||
|
||||
request.setFinish { result, error in
|
||||
timeoutTask.cancel()
|
||||
// ... 回调处理
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 4. COS 初始化队列安全
|
||||
|
||||
```swift
|
||||
private func ensureCOSInitialized(tokenData: TcTokenData) {
|
||||
guard !Self.isCOSInitialized else { return }
|
||||
|
||||
// 确保在主队列执行 COS 初始化
|
||||
if Thread.isMainThread {
|
||||
performCOSInitialization(tokenData: tokenData)
|
||||
} else {
|
||||
DispatchQueue.main.sync {
|
||||
performCOSInitialization(tokenData: tokenData)
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 5. 替换调试信息调用
|
||||
|
||||
将所有 `debugInfoSync` 调用替换为 `print`,避免异步调试信息导致的队列问题。
|
||||
|
||||
## 修复效果
|
||||
|
||||
1. **消除队列断言失败**:所有回调都在主队列执行
|
||||
2. **提高稳定性**:添加超时处理和错误恢复机制
|
||||
3. **改善调试体验**:使用同步打印避免队列冲突
|
||||
4. **保持功能完整**:所有原有功能保持不变
|
||||
|
||||
## 测试建议
|
||||
|
||||
1. 测试单张图片上传
|
||||
2. 测试多张图片批量上传
|
||||
3. 测试网络异常情况下的上传
|
||||
4. 测试大文件上传
|
||||
5. 测试并发上传场景
|
||||
|
||||
## 相关文件
|
||||
|
||||
- `yana/Utils/COSManager.swift` - 主要修复文件
|
||||
- `yana/Features/EditFeedFeature.swift` - 已正确使用 MainActor
|
||||
- `yana/Features/CreateFeedFeature.swift` - 已正确使用 MainActor
|
||||
- `yana/Features/AppSettingFeature.swift` - 已正确使用 MainActor
|
||||
|
||||
## 注意事项
|
||||
|
||||
1. 所有 UI 更新操作必须在主队列执行
|
||||
2. 腾讯云 COS 回调必须在主队列处理
|
||||
3. 避免在回调中使用异步调试信息
|
||||
4. 添加适当的超时和错误处理机制
|
||||
99
issues/多语言问题修复.md
Normal file
99
issues/多语言问题修复.md
Normal file
@@ -0,0 +1,99 @@
|
||||
# 多语言问题修复计划
|
||||
|
||||
## 问题描述
|
||||
项目配置了多语言支持,默认英文,但应用仍显示中文。原因是大部分视图使用 `NSLocalizedString`,它会读取系统语言设置而不是应用内保存的用户语言选择。
|
||||
|
||||
## 解决方案
|
||||
|
||||
### 1. 修复 LocalizationManager
|
||||
- ✅ 启用了注释的 String 扩展
|
||||
- ✅ 添加了全局 `LocalizedString` 方法
|
||||
- ✅ 添加了 `LocalizedTextModifier` 结构体
|
||||
|
||||
### 2. 替换关键界面的本地化方法
|
||||
- ✅ LoginView - 应用标题、登录按钮
|
||||
- ✅ UserAgreementView - 用户协议文本
|
||||
- ✅ FeedListView - 页面标题、空状态、标语
|
||||
- ✅ IDLoginView - 标题、占位符、按钮文本
|
||||
- ✅ EMailLoginView - 标题、按钮文本
|
||||
- ✅ LanguageSettingsView - 添加测试区域
|
||||
- ✅ MeView - 用户昵称、ID显示、加载状态、错误信息
|
||||
|
||||
### 3. 修复 MeView 显示问题
|
||||
- ✅ 修复 MainFeature 中的数据加载逻辑
|
||||
- ✅ 在 accountModelLoaded 中添加 MeView 数据加载触发
|
||||
- ✅ 确保 uid 正确设置时触发数据加载
|
||||
|
||||
### 4. 全面替换硬编码文本
|
||||
- ✅ **EditFeedView** - 上传进度提示、标题、按钮文本、占位符文本
|
||||
- ✅ **WebView** - 错误提示、操作按钮
|
||||
- ✅ **AppSettingView** - 错误提示、按钮文本、昵称限制
|
||||
- ✅ **ImagePreviewView** - 加载状态、操作按钮
|
||||
- ✅ **ImagePickerWithPreviewView** - 拍照、相册选择按钮
|
||||
- ✅ **TestView** - 测试页面文本
|
||||
- ✅ **LanguageSettingsView** - 语言设置相关文本、测试区域
|
||||
- ✅ **ConfigView** - 配置测试相关文本
|
||||
- ✅ **ScreenAdapterExample** - 示例文本
|
||||
|
||||
### 5. 修复编译错误
|
||||
- ✅ 删除重复的 ContentView.swift 文件
|
||||
- ✅ 修复 EditFeedView 中的作用域问题
|
||||
- ✅ 修复本地化字符串的调用语法
|
||||
- ✅ 确保所有变量在正确的作用域内
|
||||
|
||||
### 6. 更新本地化文件
|
||||
- ✅ 在 `en.lproj/Localizable.strings` 中添加英文翻译
|
||||
- ✅ 在 `zh-Hans.lproj/Localizable.strings` 中添加中文翻译
|
||||
- ✅ 新增 40+ 个本地化键值对
|
||||
|
||||
### 7. 新增功能
|
||||
- ✅ 全局 `LocalizedString(key, comment:)` 方法
|
||||
- ✅ String 扩展:`"key".localized`
|
||||
- ✅ 语言切换测试区域
|
||||
|
||||
## 本地化键命名规范
|
||||
- `edit_feed.*` - 编辑动态相关
|
||||
- `web_view.*` - 网页视图相关
|
||||
- `language_settings.*` - 语言设置相关
|
||||
- `app_settings.*` - 应用设置相关
|
||||
- `test.*` - 测试相关
|
||||
- `image_picker.*` - 图片选择相关
|
||||
- `content_view.*` - 内容视图相关
|
||||
- `screen_adapter.*` - 屏幕适配相关
|
||||
- `config.*` - 配置相关
|
||||
|
||||
## 使用方法
|
||||
|
||||
### 方法1:使用全局方法
|
||||
```swift
|
||||
Text(LocalizedString("login.app_title", comment: ""))
|
||||
```
|
||||
|
||||
### 方法2:使用 String 扩展
|
||||
```swift
|
||||
Text("login.app_title".localized)
|
||||
```
|
||||
|
||||
### 方法3:带参数的本地化
|
||||
```swift
|
||||
Text(LocalizedString("edit_feed.uploading_progress", comment: "").localized(arguments: Int(progress * 100)))
|
||||
```
|
||||
|
||||
## 测试验证
|
||||
1. 在语言设置界面可以看到测试区域
|
||||
2. 切换语言后,测试区域的文本会实时更新
|
||||
3. 所有使用 `LocalizedString` 的界面都会正确显示选择的语言
|
||||
4. 动态文本(进度、时间戳等)正确显示
|
||||
5. 所有硬编码文本已替换为本地化字符串
|
||||
|
||||
## 完成状态
|
||||
- ✅ 核心多语言功能修复
|
||||
- ✅ MeView 显示问题修复
|
||||
- ✅ 所有硬编码文本替换完成
|
||||
- ✅ 本地化文件更新完成
|
||||
- ✅ 测试验证通过
|
||||
|
||||
## 后续工作
|
||||
- 继续监控是否有遗漏的硬编码文本
|
||||
- 确保所有用户可见的文本都使用新的本地化方法
|
||||
- 测试各种语言切换场景
|
||||
124
issues/组件抽离到CommonComponents重构.md
Normal file
124
issues/组件抽离到CommonComponents重构.md
Normal file
@@ -0,0 +1,124 @@
|
||||
# 组件抽离到CommonComponents重构
|
||||
|
||||
## 重构概述
|
||||
|
||||
将MVVM目录中重复定义的UI组件抽离到`CommonComponents.swift`中,实现组件的统一管理和复用,避免代码重复。
|
||||
|
||||
## 重名组件分析
|
||||
|
||||
### 发现的重名组件
|
||||
1. **IDLoginBackgroundView** - 在`IDLoginPage.swift`和`IDLoginView.swift`中重复定义
|
||||
2. **IDLoginHeaderView** - 在`IDLoginPage.swift`和`IDLoginView.swift`中重复定义
|
||||
3. **CustomInputField** - 在`IDLoginPage.swift`、`IDLoginView.swift`和`CommonComponents.swift`中重复定义
|
||||
4. **IDLoginButton/IDLoginButtonView** - 在`IDLoginPage.swift`和`IDLoginView.swift`中重复定义
|
||||
|
||||
### 组件功能对比
|
||||
所有重复组件功能完全相同,只是命名略有不同,适合统一管理。
|
||||
|
||||
## 重构方案
|
||||
|
||||
### 1. 组件命名统一
|
||||
- `IDLoginBackgroundView` → `LoginBackgroundView`
|
||||
- `IDLoginHeaderView` → `LoginHeaderView`
|
||||
- `IDLoginButtonView` → `LoginButtonView`
|
||||
- `CustomInputField` → 保持原名(已在CommonComponents中)
|
||||
|
||||
### 2. 文件修改列表
|
||||
|
||||
#### 修改的文件
|
||||
- `yana/MVVM/IDLoginPage.swift` - 移除重复组件,使用CommonComponents
|
||||
- `yana/Views/IDLoginView.swift` - 移除重复组件,使用CommonComponents
|
||||
- `yana/MVVM/EMailLoginPage.swift` - 使用CommonComponents组件
|
||||
- `yana/MVVM/RecoverPasswordPage.swift` - 使用CommonComponents组件
|
||||
- `yana/MVVM/LoginPage.swift` - 使用CommonComponents组件
|
||||
- `yana/MVVM/Splash.swift` - 使用CommonComponents组件
|
||||
- `yana/MVVM/MainPage.swift` - 使用CommonComponents组件
|
||||
|
||||
#### 保持的文件
|
||||
- `yana/MVVM/CommonComponents.swift` - 统一管理所有组件
|
||||
|
||||
## 重构内容
|
||||
|
||||
### 1. IDLoginPage.swift
|
||||
- ✅ 移除`IDLoginBackgroundView`、`IDLoginHeaderView`、`CustomInputField`、`IDLoginButton`组件定义
|
||||
- ✅ 使用`LoginBackgroundView`、`LoginHeaderView`、`CustomInputField`、`LoginButtonView`
|
||||
- ✅ 保持ViewModel和主视图逻辑不变
|
||||
|
||||
### 2. IDLoginView.swift (Views目录)
|
||||
- ✅ 移除`IDLoginBackgroundView`、`IDLoginHeaderView`、`CustomInputField`、`IDLoginButtonView`组件定义
|
||||
- ✅ 使用`LoginBackgroundView`、`LoginHeaderView`、`CustomInputField`、`LoginButtonView`
|
||||
- ✅ 保持TCA架构和主视图逻辑不变
|
||||
|
||||
### 3. EMailLoginPage.swift
|
||||
- ✅ 使用`LoginBackgroundView`替换直接使用`Image("bg")`
|
||||
- ✅ 使用`LoginHeaderView`替换内联的导航栏代码
|
||||
- ✅ 使用`LoginButtonView`替换内联的按钮代码
|
||||
- ✅ 使用`CustomInputField`替换内联的输入框代码
|
||||
- ✅ 简化了UI组件的定义,提高代码复用性
|
||||
|
||||
### 4. RecoverPasswordPage.swift
|
||||
- ✅ 使用`LoginBackgroundView`替换直接使用`Image("bg")`
|
||||
- ✅ 使用`LoginHeaderView`替换内联的导航栏代码
|
||||
- ✅ 保持其他UI组件不变(因为它们是特定的)
|
||||
|
||||
### 5. LoginPage.swift
|
||||
- ✅ 使用`LoginBackgroundView`替换`backgroundView`中的`Image("bg")`
|
||||
- ✅ 保持其他特定组件不变
|
||||
|
||||
### 6. Splash.swift
|
||||
- ✅ 使用`LoginBackgroundView`替换`Image("bg")`
|
||||
- ✅ 保持启动画面的其他元素不变
|
||||
|
||||
### 7. MainPage.swift
|
||||
- ✅ 使用`LoginBackgroundView`替换`Image("bg")`
|
||||
- ✅ 保持底部导航栏等特定组件不变
|
||||
|
||||
## 技术要点
|
||||
|
||||
### 1. 组件接口保持兼容
|
||||
- 所有组件的参数和返回值保持不变
|
||||
- 确保现有调用代码无需修改
|
||||
|
||||
### 2. 命名规范统一
|
||||
- 使用通用的`Login`前缀,而不是特定的`IDLogin`前缀
|
||||
- 保持组件名称的语义清晰
|
||||
|
||||
### 3. 代码复用最大化
|
||||
- 背景图片、导航栏、按钮等通用组件统一管理
|
||||
- 输入框组件支持多种类型(text、number、password、verificationCode)
|
||||
|
||||
## 验证结果
|
||||
|
||||
### 组件定义验证
|
||||
- ✅ `LoginBackgroundView` - 仅在CommonComponents中定义
|
||||
- ✅ `LoginHeaderView` - 仅在CommonComponents中定义
|
||||
- ✅ `LoginButtonView` - 仅在CommonComponents中定义
|
||||
- ✅ `CustomInputField` - 仅在CommonComponents中定义
|
||||
|
||||
### 组件使用验证
|
||||
- ✅ 所有MVVM文件都正确使用了CommonComponents中的组件
|
||||
- ✅ 没有发现重复的组件定义
|
||||
- ✅ 组件调用接口保持一致
|
||||
|
||||
### 功能验证
|
||||
- ✅ 所有页面的UI显示正常
|
||||
- ✅ 组件交互功能正常
|
||||
- ✅ 没有引入新的编译错误
|
||||
|
||||
## 后续优化建议
|
||||
|
||||
1. **组件扩展**:可以考虑将更多通用组件添加到CommonComponents中
|
||||
2. **主题支持**:为组件添加主题支持,支持不同的颜色方案
|
||||
3. **动画支持**:为组件添加统一的动画效果
|
||||
4. **无障碍支持**:为组件添加无障碍标签和描述
|
||||
5. **测试覆盖**:为CommonComponents中的组件添加单元测试
|
||||
6. **文档完善**:为每个组件添加详细的使用文档和示例
|
||||
|
||||
## 完成状态
|
||||
- ✅ 重名组件识别和分析
|
||||
- ✅ 组件抽离到CommonComponents
|
||||
- ✅ 所有MVVM文件更新完成
|
||||
- ✅ Views目录文件更新完成
|
||||
- ✅ 组件使用验证通过
|
||||
- ✅ 功能验证通过
|
||||
- ✅ 文档记录完成
|
||||
16
ui-demo.swift
Normal file
16
ui-demo.swift
Normal file
@@ -0,0 +1,16 @@
|
||||
let label = UILabel()
|
||||
let attrString = NSMutableAttributedString(string: "Agree to the "User Service Agreement" and "Privacy Policy"")
|
||||
label.frame = CGRect(x: 71, y: 735, width: 256, height: 34)
|
||||
label.numberOfLines = 0
|
||||
let attr: [NSAttributedString.Key : Any] = [.font: UIFont(name: "PingFang SC-Regular", size: 12),.foregroundColor: UIColor(red: 1, green: 1, blue: 1, alpha: 1)]
|
||||
attrString.addAttributes(attr, range: NSRange(location: 0, length: attrString.length))
|
||||
view.addSubview(label)
|
||||
let strSubAttr1: [NSMutableAttributedString.Key: Any] = [.font: UIFont(name: "PingFang SC-Regular", size: 12),.foregroundColor: UIColor(red: 1, green: 1, blue: 1,alpha:1)]
|
||||
attrString.addAttributes(strSubAttr1, range: NSRange(location: 0, length: 13))
|
||||
let strSubAttr2: [NSMutableAttributedString.Key: Any] = [.font: UIFont(name: "PingFang SC-Regular", size: 12),.foregroundColor: UIColor(red: 0.78, green: 0.35, blue: 1,alpha:1)]
|
||||
attrString.addAttributes(strSubAttr2, range: NSRange(location: 13, length: 24))
|
||||
let strSubAttr3: [NSMutableAttributedString.Key: Any] = [.font: UIFont(name: "PingFang SC-Regular", size: 12),.foregroundColor: UIColor(red: 1, green: 1, blue: 1,alpha:1)]
|
||||
attrString.addAttributes(strSubAttr3, range: NSRange(location: 37, length: 5))
|
||||
let strSubAttr4: [NSMutableAttributedString.Key: Any] = [.font: UIFont(name: "PingFang SC-Regular", size: 12),.foregroundColor: UIColor(red: 0.78, green: 0.35, blue: 1,alpha:1)]
|
||||
attrString.addAttributes(strSubAttr4, range: NSRange(location: 42, length: 16))
|
||||
label.attributedText = attrString
|
||||
@@ -12,6 +12,7 @@
|
||||
4C4C91302DE864F000384527 /* ComposableArchitecture in Frameworks */ = {isa = PBXBuildFile; productRef = 4C4C912F2DE864F000384527 /* ComposableArchitecture */; };
|
||||
4CE9EFEA2E28FC3B0078D046 /* CasePaths in Frameworks */ = {isa = PBXBuildFile; productRef = 4CE9EFE92E28FC3B0078D046 /* CasePaths */; };
|
||||
4CE9EFEC2E28FC3B0078D046 /* CasePathsCore in Frameworks */ = {isa = PBXBuildFile; productRef = 4CE9EFEB2E28FC3B0078D046 /* CasePathsCore */; };
|
||||
4CFE5EBA2E38E8D400836B0C /* Atomics in Frameworks */ = {isa = PBXBuildFile; productRef = 4CFE5EB92E38E8D400836B0C /* Atomics */; };
|
||||
DF26704F4C1F2ABA0405A0C3 /* Pods_yana.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = E9DBA38268C435BE06F39AE2 /* Pods_yana.framework */; };
|
||||
/* End PBXBuildFile section */
|
||||
|
||||
@@ -73,6 +74,7 @@
|
||||
4C4C91302DE864F000384527 /* ComposableArchitecture in Frameworks */,
|
||||
4C4C910B2DE85A4F00384527 /* CoreFoundation.framework in Frameworks */,
|
||||
DF26704F4C1F2ABA0405A0C3 /* Pods_yana.framework in Frameworks */,
|
||||
4CFE5EBA2E38E8D400836B0C /* Atomics in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
@@ -215,6 +217,7 @@
|
||||
packageReferences = (
|
||||
4C4C912E2DE864F000384527 /* XCRemoteSwiftPackageReference "swift-composable-architecture" */,
|
||||
4CE9EFE82E28FC3B0078D046 /* XCRemoteSwiftPackageReference "swift-case-paths" */,
|
||||
4CFE5EB82E38E8D400836B0C /* XCRemoteSwiftPackageReference "swift-atomics" */,
|
||||
);
|
||||
preferredProjectObjectVersion = 77;
|
||||
productRefGroup = 4C3E65202DB61F7A00E5A455 /* Products */;
|
||||
@@ -376,7 +379,7 @@
|
||||
DEVELOPMENT_TEAM = EKM7RAGNA6;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_TESTABILITY = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = NO;
|
||||
GCC_C_LANGUAGE_STANDARD = gnu17;
|
||||
GCC_DYNAMIC_NO_PIC = NO;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
@@ -391,7 +394,7 @@
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 18.4;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
|
||||
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
|
||||
MTL_FAST_MATH = YES;
|
||||
@@ -442,7 +445,7 @@
|
||||
DEVELOPMENT_TEAM = EKM7RAGNA6;
|
||||
ENABLE_NS_ASSERTIONS = NO;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = NO;
|
||||
GCC_C_LANGUAGE_STANDARD = gnu17;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
@@ -451,7 +454,7 @@
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 18.4;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
MTL_FAST_MATH = YES;
|
||||
@@ -499,7 +502,7 @@
|
||||
INFOPLIST_KEY_UILaunchStoryboardName = LaunchScreen;
|
||||
INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait;
|
||||
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown";
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 16.0;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
@@ -557,7 +560,7 @@
|
||||
INFOPLIST_KEY_UILaunchStoryboardName = LaunchScreen;
|
||||
INFOPLIST_KEY_UISupportedInterfaceOrientations = UIInterfaceOrientationPortrait;
|
||||
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown";
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 16.0;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
@@ -588,7 +591,7 @@
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
DEVELOPMENT_TEAM = EKM7RAGNA6;
|
||||
GENERATE_INFOPLIST_FILE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 16.6;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
MARKETING_VERSION = 1.0;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.junpeiqi.eparty;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
@@ -612,7 +615,7 @@
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
DEVELOPMENT_TEAM = EKM7RAGNA6;
|
||||
GENERATE_INFOPLIST_FILE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 16.6;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.6;
|
||||
MARKETING_VERSION = 1.0;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.stupidmonkey.yanaAPITests;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
@@ -676,6 +679,14 @@
|
||||
kind = branch;
|
||||
};
|
||||
};
|
||||
4CFE5EB82E38E8D400836B0C /* XCRemoteSwiftPackageReference "swift-atomics" */ = {
|
||||
isa = XCRemoteSwiftPackageReference;
|
||||
repositoryURL = "https://github.com/apple/swift-atomics.git";
|
||||
requirement = {
|
||||
kind = upToNextMajorVersion;
|
||||
minimumVersion = 1.3.0;
|
||||
};
|
||||
};
|
||||
/* End XCRemoteSwiftPackageReference section */
|
||||
|
||||
/* Begin XCSwiftPackageProductDependency section */
|
||||
@@ -694,6 +705,11 @@
|
||||
package = 4CE9EFE82E28FC3B0078D046 /* XCRemoteSwiftPackageReference "swift-case-paths" */;
|
||||
productName = CasePathsCore;
|
||||
};
|
||||
4CFE5EB92E38E8D400836B0C /* Atomics */ = {
|
||||
isa = XCSwiftPackageProductDependency;
|
||||
package = 4CFE5EB82E38E8D400836B0C /* XCRemoteSwiftPackageReference "swift-atomics" */;
|
||||
productName = Atomics;
|
||||
};
|
||||
/* End XCSwiftPackageProductDependency section */
|
||||
};
|
||||
rootObject = 4C3E65172DB61F7A00E5A455 /* Project object */;
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"originHash" : "411c4947a4ddec377a0aac37852b26ccdf5b2dd58cb99bedfb2d4c108efd60fd",
|
||||
"originHash" : "ee5640a3641e5c53e0d4d0295dacfe48036738ce817585081693672ac6a81318",
|
||||
"pins" : [
|
||||
{
|
||||
"identity" : "combine-schedulers",
|
||||
@@ -10,6 +10,15 @@
|
||||
"version" : "1.0.3"
|
||||
}
|
||||
},
|
||||
{
|
||||
"identity" : "swift-atomics",
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/apple/swift-atomics.git",
|
||||
"state" : {
|
||||
"revision" : "b601256eab081c0f92f059e12818ac1d4f178ff7",
|
||||
"version" : "1.3.0"
|
||||
}
|
||||
},
|
||||
{
|
||||
"identity" : "swift-case-paths",
|
||||
"kind" : "remoteSourceControl",
|
||||
@@ -33,8 +42,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/apple/swift-collections",
|
||||
"state" : {
|
||||
"revision" : "c1805596154bb3a265fd91b8ac0c4433b4348fb0",
|
||||
"version" : "1.2.0"
|
||||
"revision" : "8c0c0a8b49e080e54e5e328cc552821ff07cd341",
|
||||
"version" : "1.2.1"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -42,8 +51,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/swift-composable-architecture",
|
||||
"state" : {
|
||||
"revision" : "6574de2396319a58e86e2178577268cb4aeccc30",
|
||||
"version" : "1.20.2"
|
||||
"revision" : "4c47829a080789cf20d82c64d8c27291352391d4",
|
||||
"version" : "1.21.1"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -69,8 +78,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/swift-dependencies",
|
||||
"state" : {
|
||||
"revision" : "4c90d6b2b9bf0911af87b103bb40f41771891596",
|
||||
"version" : "1.9.2"
|
||||
"revision" : "eefcdaa88d2e2fd82e3405dfb6eb45872011a0b5",
|
||||
"version" : "1.9.3"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -87,8 +96,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/swift-navigation",
|
||||
"state" : {
|
||||
"revision" : "db6bc9dbfed001f21e6728fd36413d9342c235b4",
|
||||
"version" : "2.3.0"
|
||||
"revision" : "4e89284c1966538109dc783497405bc680e9bc96",
|
||||
"version" : "2.4.0"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -96,8 +105,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/swift-perception",
|
||||
"state" : {
|
||||
"revision" : "d924c62a70fca5f43872f286dbd7cef0957f1c01",
|
||||
"version" : "1.6.0"
|
||||
"revision" : "328a0b49e2690135c4c2660661f0ed83f16853e3",
|
||||
"version" : "2.0.4"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -105,8 +114,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/swift-sharing",
|
||||
"state" : {
|
||||
"revision" : "75e846ee3159dc75b3a29bfc24b6ce5a557ddca9",
|
||||
"version" : "2.5.2"
|
||||
"revision" : "5d87dda90ed048f216826efbad404110141161bb",
|
||||
"version" : "2.6.0"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -123,8 +132,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/pointfreeco/xctest-dynamic-overlay",
|
||||
"state" : {
|
||||
"revision" : "39de59b2d47f7ef3ca88a039dff3084688fe27f4",
|
||||
"version" : "1.5.2"
|
||||
"revision" : "23e3442166b5122f73f9e3e622cd1e4bafeab3b7",
|
||||
"version" : "1.6.0"
|
||||
}
|
||||
}
|
||||
],
|
||||
|
||||
@@ -63,6 +63,13 @@
|
||||
ReferencedContainer = "container:yana.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
<EnvironmentVariables>
|
||||
<EnvironmentVariable
|
||||
key = "OS_ACTIVITY_MODE"
|
||||
value = "disable"
|
||||
isEnabled = "NO">
|
||||
</EnvironmentVariable>
|
||||
</EnvironmentVariables>
|
||||
</LaunchAction>
|
||||
<ProfileAction
|
||||
buildConfiguration = "Release"
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"originHash" : "411c4947a4ddec377a0aac37852b26ccdf5b2dd58cb99bedfb2d4c108efd60fd",
|
||||
"originHash" : "d23aef0dd86826b19606675a068b14e16000420ac169efa6217629c0ab2b0f5f",
|
||||
"pins" : [
|
||||
{
|
||||
"identity" : "combine-schedulers",
|
||||
@@ -10,6 +10,15 @@
|
||||
"version" : "1.0.3"
|
||||
}
|
||||
},
|
||||
{
|
||||
"identity" : "swift-atomics",
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/apple/swift-atomics.git",
|
||||
"state" : {
|
||||
"revision" : "b601256eab081c0f92f059e12818ac1d4f178ff7",
|
||||
"version" : "1.3.0"
|
||||
}
|
||||
},
|
||||
{
|
||||
"identity" : "swift-case-paths",
|
||||
"kind" : "remoteSourceControl",
|
||||
|
||||
@@ -16,7 +16,7 @@
|
||||
| 环境 | 地址 | 说明 |
|
||||
|------|------|------|
|
||||
| 生产环境 | `https://api.epartylive.com` | 正式服务器 |
|
||||
| 测试环境 | `http://beta.api.molistar.xyz` | 开发测试服务器 |
|
||||
| 测试环境 | `http://beta.api.pekolive.com` | 开发测试服务器 |
|
||||
| 图片服务 | `https://image.hfighting.com` | 静态资源服务器 |
|
||||
|
||||
**环境切换机制:**
|
||||
|
||||
@@ -25,10 +25,13 @@ enum APIEndpoint: String, CaseIterable {
|
||||
case getUserInfo = "/user/get" // 新增:获取用户信息端点
|
||||
case getMyDynamic = "/dynamic/getMyDynamic"
|
||||
case updateUser = "/user/v2/update" // 新增:用户信息更新端点
|
||||
case dynamicLike = "/dynamic/like" // 新增:动态点赞/取消点赞端点
|
||||
case deleteDynamic = "/dynamic/delete" // 新增:删除动态端点
|
||||
|
||||
// Web 页面路径
|
||||
case userAgreement = "/modules/rule/protocol.html"
|
||||
case privacyPolicy = "/modules/rule/privacy-wap.html"
|
||||
case deactivateAccount = "/modules/logout/confirm.html"
|
||||
|
||||
|
||||
var path: String {
|
||||
@@ -99,7 +102,7 @@ struct APIConfiguration {
|
||||
"Accept-Encoding": "gzip, br",
|
||||
"Accept-Language": Locale.current.language.languageCode?.identifier ?? "en",
|
||||
"App-Version": Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String ?? "1.0.0",
|
||||
"User-Agent": "YuMi/20.20.61 (iPhone; iOS 16.4; Scale/2.00)"
|
||||
"User-Agent": await UserAgentProvider.userAgent()
|
||||
]
|
||||
// 检查用户认证状态并添加相关 headers
|
||||
let authStatus = await UserInfoManager.checkAuthenticationStatus()
|
||||
|
||||
@@ -1,218 +1,286 @@
|
||||
import Foundation
|
||||
|
||||
// MARK: - API Logger
|
||||
@MainActor
|
||||
class APILogger {
|
||||
enum LogLevel {
|
||||
case none
|
||||
case basic
|
||||
case detailed
|
||||
}
|
||||
|
||||
// 使用 actor 封装可变全局状态以保证并发安全
|
||||
actor Config {
|
||||
static let shared = Config()
|
||||
#if DEBUG
|
||||
private var level: LogLevel = .detailed
|
||||
#else
|
||||
private var level: LogLevel = .none
|
||||
#endif
|
||||
func get() -> LogLevel { level }
|
||||
func set(_ newLevel: LogLevel) { level = newLevel }
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
static var logLevel: LogLevel = .detailed
|
||||
#else
|
||||
static var logLevel: LogLevel = .none
|
||||
#endif
|
||||
|
||||
private static let logQueue = DispatchQueue(label: "com.yana.api.logger", qos: .utility)
|
||||
|
||||
private static let dateFormatter: DateFormatter = {
|
||||
let formatter = DateFormatter()
|
||||
formatter.dateFormat = "HH:mm:ss.SSS"
|
||||
return formatter
|
||||
}()
|
||||
|
||||
// MARK: - Redaction
|
||||
/// 需要脱敏的敏感字段(统一小写匹配)
|
||||
private static let sensitiveKeys: Set<String> = [
|
||||
"authorization", "token", "access-token", "access_token", "refresh-token", "refresh_token",
|
||||
"password", "passwd", "secret", "pub_ticket", "ticket", "set-cookie", "cookie"
|
||||
]
|
||||
/// 对字符串做中间遮罩,保留前后若干字符
|
||||
private static func maskString(_ value: String, keepPrefix: Int = 3, keepSuffix: Int = 2) -> String {
|
||||
guard !value.isEmpty else { return value }
|
||||
if value.count <= keepPrefix + keepSuffix { return String(repeating: "*", count: value.count) }
|
||||
let start = value.startIndex
|
||||
let prefixEnd = value.index(start, offsetBy: keepPrefix)
|
||||
let suffixStart = value.index(value.endIndex, offsetBy: -keepSuffix)
|
||||
let prefix = value[start..<prefixEnd]
|
||||
let suffix = value[suffixStart..<value.endIndex]
|
||||
return String(prefix) + String(repeating: "*", count: max(0, value.distance(from: prefixEnd, to: suffixStart))) + String(suffix)
|
||||
}
|
||||
/// 对 headers 进行脱敏
|
||||
private static func maskHeaders(_ headers: [String: String]) -> [String: String] {
|
||||
var masked: [String: String] = [:]
|
||||
for (key, value) in headers {
|
||||
if sensitiveKeys.contains(key.lowercased()) {
|
||||
masked[key] = maskString(value)
|
||||
} else {
|
||||
masked[key] = value
|
||||
}
|
||||
}
|
||||
return masked
|
||||
}
|
||||
/// 递归地对 JSON 对象进行脱敏
|
||||
private static func redactJSONObject(_ obj: Any) -> Any {
|
||||
if let dict = obj as? [String: Any] {
|
||||
var newDict: [String: Any] = [:]
|
||||
for (k, v) in dict {
|
||||
if sensitiveKeys.contains(k.lowercased()) {
|
||||
if let str = v as? String { newDict[k] = maskString(str) }
|
||||
else { newDict[k] = "<redacted>" }
|
||||
} else {
|
||||
newDict[k] = redactJSONObject(v)
|
||||
}
|
||||
}
|
||||
return newDict
|
||||
} else if let arr = obj as? [Any] {
|
||||
return arr.map { redactJSONObject($0) }
|
||||
} else {
|
||||
return obj
|
||||
}
|
||||
}
|
||||
/// 将请求体 Data 以 Pretty JSON(脱敏后)或摘要形式输出
|
||||
private static func maskedBodyString(from body: Data?) -> String {
|
||||
guard let body = body, !body.isEmpty else { return "No body" }
|
||||
if let json = try? JSONSerialization.jsonObject(with: body, options: []) {
|
||||
let redacted = redactJSONObject(json)
|
||||
if let pretty = try? JSONSerialization.data(withJSONObject: redacted, options: [.prettyPrinted]),
|
||||
let prettyString = String(data: pretty, encoding: .utf8) {
|
||||
return prettyString
|
||||
}
|
||||
}
|
||||
return "<non-json body> (\(body.count) bytes)"
|
||||
}
|
||||
|
||||
// MARK: - Request Logging
|
||||
@MainActor static func logRequest<T: APIRequestProtocol>(
|
||||
static func logRequest<T: APIRequestProtocol>(
|
||||
_ request: T,
|
||||
url: URL,
|
||||
body: Data?,
|
||||
finalHeaders: [String: String]? = nil
|
||||
) {
|
||||
#if DEBUG
|
||||
guard logLevel != .none else { return }
|
||||
#else
|
||||
#if !DEBUG
|
||||
return
|
||||
#endif
|
||||
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
|
||||
print("\n🚀 [API Request] [\(timestamp)] ==================")
|
||||
print("📍 Endpoint: \(request.endpoint)")
|
||||
print("🔗 Full URL: \(url.absoluteString)")
|
||||
print("📝 Method: \(request.method.rawValue)")
|
||||
print("⏰ Timeout: \(request.timeout)s")
|
||||
|
||||
// 显示最终的完整 headers(包括默认 headers 和自定义 headers)
|
||||
if let headers = finalHeaders, !headers.isEmpty {
|
||||
if logLevel == .detailed {
|
||||
print("📋 Final Headers (包括默认 + 自定义):")
|
||||
for (key, value) in headers.sorted(by: { $0.key < $1.key }) {
|
||||
print(" \(key): \(value)")
|
||||
}
|
||||
} else if logLevel == .basic {
|
||||
print("📋 Headers: \(headers.count) 个 headers")
|
||||
// 只显示重要的 headers
|
||||
let importantHeaders = ["Content-Type", "Accept", "User-Agent", "Authorization"]
|
||||
for key in importantHeaders {
|
||||
if let value = headers[key] {
|
||||
print(" \(key): \(value)")
|
||||
#else
|
||||
Task {
|
||||
let level = await Config.shared.get()
|
||||
guard level != .none else { return }
|
||||
logQueue.async {
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
debugInfoSync("\n🚀 [API Request] [\(timestamp)] ==================")
|
||||
debugInfoSync("📍 Endpoint: \(request.endpoint)")
|
||||
debugInfoSync("🔗 Full URL: \(url.absoluteString)")
|
||||
debugInfoSync("📝 Method: \(request.method.rawValue)")
|
||||
debugInfoSync("⏰ Timeout: \(request.timeout)s")
|
||||
|
||||
// 显示最终的完整 headers(包括默认 headers 和自定义 headers)
|
||||
if let headers = finalHeaders, !headers.isEmpty {
|
||||
if level == .detailed {
|
||||
debugInfoSync("📋 Final Headers (包括默认 + 自定义):")
|
||||
let masked = maskHeaders(headers)
|
||||
for (key, value) in masked.sorted(by: { $0.key < $1.key }) {
|
||||
debugInfoSync(" \(key): \(value)")
|
||||
}
|
||||
} else if level == .basic {
|
||||
debugInfoSync("📋 Headers: \(headers.count) 个 headers")
|
||||
// 只显示重要的 headers
|
||||
let importantHeaders = ["Content-Type", "Accept", "User-Agent", "Authorization"]
|
||||
let masked = maskHeaders(headers)
|
||||
for key in importantHeaders {
|
||||
if let value = masked[key] {
|
||||
debugInfoSync(" \(key): \(value)")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if let customHeaders = request.headers, !customHeaders.isEmpty {
|
||||
print("📋 Custom Headers:")
|
||||
for (key, value) in customHeaders.sorted(by: { $0.key < $1.key }) {
|
||||
print(" \(key): \(value)")
|
||||
}
|
||||
} else {
|
||||
print("📋 Headers: 使用默认 headers")
|
||||
}
|
||||
|
||||
if let queryParams = request.queryParameters, !queryParams.isEmpty {
|
||||
print("🔍 Query Parameters:")
|
||||
for (key, value) in queryParams.sorted(by: { $0.key < $1.key }) {
|
||||
print(" \(key): \(value)")
|
||||
}
|
||||
}
|
||||
|
||||
if logLevel == .detailed {
|
||||
if let body = body {
|
||||
print("📦 Request Body (\(body.count) bytes):")
|
||||
if let jsonObject = try? JSONSerialization.jsonObject(with: body, options: []),
|
||||
let prettyData = try? JSONSerialization.data(withJSONObject: jsonObject, options: .prettyPrinted),
|
||||
let prettyString = String(data: prettyData, encoding: .utf8) {
|
||||
print(prettyString)
|
||||
} else if let rawString = String(data: body, encoding: .utf8) {
|
||||
print(rawString)
|
||||
} else {
|
||||
print("Binary data")
|
||||
} else if let customHeaders = request.headers, !customHeaders.isEmpty {
|
||||
debugInfoSync("📋 Custom Headers:")
|
||||
let masked = maskHeaders(customHeaders)
|
||||
for (key, value) in masked.sorted(by: { $0.key < $1.key }) {
|
||||
debugInfoSync(" \(key): \(value)")
|
||||
}
|
||||
} else {
|
||||
print("📦 Request Body: No body")
|
||||
debugInfoSync("📋 Headers: 使用默认 headers")
|
||||
}
|
||||
|
||||
// 显示基础参数信息(仅详细模式)
|
||||
if request.includeBaseParameters {
|
||||
print("📱 Base Parameters: 自动注入设备和应用信息")
|
||||
let baseParams = BaseRequest()
|
||||
print(" Device: \(baseParams.model), OS: \(baseParams.os) \(baseParams.osVersion)")
|
||||
print(" App: \(baseParams.app) v\(baseParams.appVersion)")
|
||||
print(" Language: \(baseParams.acceptLanguage)")
|
||||
}
|
||||
} else if logLevel == .basic {
|
||||
if let body = body {
|
||||
print("📦 Request Body: \(formatBytes(body.count))")
|
||||
} else {
|
||||
print("📦 Request Body: No body")
|
||||
if let queryParams = request.queryParameters, !queryParams.isEmpty {
|
||||
debugInfoSync("🔍 Query Parameters:")
|
||||
for (key, value) in queryParams.sorted(by: { $0.key < $1.key }) {
|
||||
let masked = sensitiveKeys.contains(key.lowercased()) ? maskString(value) : value
|
||||
debugInfoSync(" \(key): \(masked)")
|
||||
}
|
||||
}
|
||||
|
||||
// 基础模式也显示是否包含基础参数
|
||||
if request.includeBaseParameters {
|
||||
print("📱 Base Parameters: 已自动注入")
|
||||
if level == .detailed {
|
||||
let pretty = maskedBodyString(from: body)
|
||||
debugInfoSync("📦 Request Body: \n\(pretty)")
|
||||
|
||||
// 仅提示包含基础参数,避免跨 actor 读取 UIKit 信息
|
||||
if request.includeBaseParameters {
|
||||
debugInfoSync("📱 Base Parameters: 已自动注入")
|
||||
}
|
||||
} else if level == .basic {
|
||||
let size = body?.count ?? 0
|
||||
debugInfoSync("📦 Request Body: \(formatBytes(size))")
|
||||
|
||||
// 基础模式也显示是否包含基础参数
|
||||
if request.includeBaseParameters {
|
||||
debugInfoSync("📱 Base Parameters: 已自动注入")
|
||||
}
|
||||
}
|
||||
debugInfoSync("=====================================")
|
||||
}
|
||||
}
|
||||
|
||||
print("=====================================")
|
||||
#endif
|
||||
}
|
||||
|
||||
// MARK: - Response Logging
|
||||
static func logResponse(data: Data, response: HTTPURLResponse, duration: TimeInterval) {
|
||||
#if DEBUG
|
||||
guard logLevel != .none else { return }
|
||||
#else
|
||||
#if !DEBUG
|
||||
return
|
||||
#endif
|
||||
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
let statusEmoji = response.statusCode < 400 ? "✅" : "❌"
|
||||
|
||||
print("\n\(statusEmoji) [API Response] [\(timestamp)] ===================")
|
||||
print("⏱️ Duration: \(String(format: "%.3f", duration))s")
|
||||
print("📊 Status Code: \(response.statusCode)")
|
||||
print("🔗 URL: \(response.url?.absoluteString ?? "Unknown")")
|
||||
print("📏 Data Size: \(formatBytes(data.count))")
|
||||
|
||||
if logLevel == .detailed {
|
||||
print("📋 Response Headers:")
|
||||
for (key, value) in response.allHeaderFields.sorted(by: { "\($0.key)" < "\($1.key)" }) {
|
||||
print(" \(key): \(value)")
|
||||
}
|
||||
#else
|
||||
Task {
|
||||
let level = await Config.shared.get()
|
||||
guard level != .none else { return }
|
||||
logQueue.async {
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
let statusEmoji = response.statusCode < 400 ? "✅" : "❌"
|
||||
debugInfoSync("\n\(statusEmoji) [API Response] [\(timestamp)] ===================")
|
||||
debugInfoSync("⏱️ Duration: \(String(format: "%.3f", duration))s")
|
||||
debugInfoSync("📊 Status Code: \(response.statusCode)")
|
||||
debugInfoSync("🔗 URL: \(response.url?.absoluteString ?? "Unknown")")
|
||||
debugInfoSync("📏 Data Size: \(formatBytes(data.count))")
|
||||
|
||||
print("📦 Response Data:")
|
||||
if data.isEmpty {
|
||||
print(" Empty response")
|
||||
} else if let jsonObject = try? JSONSerialization.jsonObject(with: data, options: []),
|
||||
let prettyData = try? JSONSerialization.data(withJSONObject: jsonObject, options: .prettyPrinted),
|
||||
let prettyString = String(data: prettyData, encoding: .utf8) {
|
||||
print(prettyString)
|
||||
} else if let rawString = String(data: data, encoding: .utf8) {
|
||||
print(rawString)
|
||||
} else {
|
||||
print(" Binary data (\(data.count) bytes)")
|
||||
if level == .detailed {
|
||||
debugInfoSync("📋 Response Headers:")
|
||||
// 将 headers 转为 [String:String] 后脱敏
|
||||
var headers: [String: String] = [:]
|
||||
for (k, v) in response.allHeaderFields { headers["\(k)"] = "\(v)" }
|
||||
let masked = maskHeaders(headers)
|
||||
for (key, value) in masked.sorted(by: { $0.key < $1.key }) {
|
||||
debugInfoSync(" \(key): \(value)")
|
||||
}
|
||||
|
||||
debugInfoSync("📦 Response Data:")
|
||||
if data.isEmpty {
|
||||
debugInfoSync(" Empty response")
|
||||
} else if let jsonObject = try? JSONSerialization.jsonObject(with: data, options: []),
|
||||
let prettyData = try? JSONSerialization.data(withJSONObject: redactJSONObject(jsonObject), options: .prettyPrinted),
|
||||
let prettyString = String(data: prettyData, encoding: .utf8) {
|
||||
debugInfoSync(prettyString)
|
||||
} else if let _ = String(data: data, encoding: .utf8) {
|
||||
// 对非 JSON 文本响应不做内容回显,避免泄漏
|
||||
debugInfoSync("<non-json text> (\(data.count) bytes)")
|
||||
} else {
|
||||
debugInfoSync(" Binary data (\(data.count) bytes)")
|
||||
}
|
||||
}
|
||||
debugInfoSync("=====================================")
|
||||
}
|
||||
}
|
||||
|
||||
print("=====================================")
|
||||
#endif
|
||||
}
|
||||
|
||||
// MARK: - Error Logging
|
||||
static func logError(_ error: Error, url: URL?, duration: TimeInterval) {
|
||||
#if DEBUG
|
||||
guard logLevel != .none else { return }
|
||||
#else
|
||||
#if !DEBUG
|
||||
return
|
||||
#endif
|
||||
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
|
||||
print("\n❌ [API Error] [\(timestamp)] ======================")
|
||||
print("⏱️ Duration: \(String(format: "%.3f", duration))s")
|
||||
if let url = url {
|
||||
print("🔗 URL: \(url.absoluteString)")
|
||||
}
|
||||
|
||||
if let apiError = error as? APIError {
|
||||
print("🚨 API Error: \(apiError.localizedDescription)")
|
||||
} else {
|
||||
print("🚨 System Error: \(error.localizedDescription)")
|
||||
}
|
||||
|
||||
if logLevel == .detailed {
|
||||
if let urlError = error as? URLError {
|
||||
print("🔍 URLError Code: \(urlError.code.rawValue)")
|
||||
print("🔍 URLError Localized: \(urlError.localizedDescription)")
|
||||
|
||||
// 详细的网络错误分析
|
||||
switch urlError.code {
|
||||
case .timedOut:
|
||||
print("💡 建议:检查网络连接或增加超时时间")
|
||||
case .notConnectedToInternet:
|
||||
print("💡 建议:检查网络连接")
|
||||
case .cannotConnectToHost:
|
||||
print("💡 建议:检查服务器地址和端口")
|
||||
case .resourceUnavailable:
|
||||
print("💡 建议:检查 API 端点是否正确")
|
||||
default:
|
||||
break
|
||||
}
|
||||
#else
|
||||
Task {
|
||||
let level = await Config.shared.get()
|
||||
guard level != .none else { return }
|
||||
logQueue.async {
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
debugErrorSync("\n❌ [API Error] [\(timestamp)] ======================")
|
||||
debugErrorSync("⏱️ Duration: \(String(format: "%.3f", duration))s")
|
||||
if let url = url {
|
||||
debugErrorSync("🔗 URL: \(url.absoluteString)")
|
||||
}
|
||||
|
||||
if let apiError = error as? APIError {
|
||||
debugErrorSync("🚨 API Error: \(apiError.localizedDescription)")
|
||||
} else {
|
||||
debugErrorSync("🚨 System Error: \(error.localizedDescription)")
|
||||
}
|
||||
|
||||
if level == .detailed {
|
||||
if let urlError = error as? URLError {
|
||||
debugInfoSync("🔍 URLError Code: \(urlError.code.rawValue)")
|
||||
debugInfoSync("🔍 URLError Localized: \(urlError.localizedDescription)")
|
||||
|
||||
// 详细的网络错误分析
|
||||
switch urlError.code {
|
||||
case .timedOut:
|
||||
debugWarnSync("💡 建议:检查网络连接或增加超时时间")
|
||||
case .notConnectedToInternet:
|
||||
debugWarnSync("💡 建议:检查网络连接")
|
||||
case .cannotConnectToHost:
|
||||
debugWarnSync("💡 建议:检查服务器地址和端口")
|
||||
case .resourceUnavailable:
|
||||
debugWarnSync("💡 建议:检查 API 端点是否正确")
|
||||
default:
|
||||
break
|
||||
}
|
||||
}
|
||||
debugInfoSync("🔍 Full Error: \(error)")
|
||||
}
|
||||
debugErrorSync("=====================================\n")
|
||||
}
|
||||
print("🔍 Full Error: \(error)")
|
||||
}
|
||||
|
||||
print("=====================================\n")
|
||||
#endif
|
||||
}
|
||||
|
||||
// MARK: - Decoded Response Logging
|
||||
static func logDecodedResponse<T>(_ response: T, type: T.Type) {
|
||||
#if DEBUG
|
||||
guard logLevel == .detailed else { return }
|
||||
#else
|
||||
#if !DEBUG
|
||||
return
|
||||
#else
|
||||
Task {
|
||||
let level = await Config.shared.get()
|
||||
guard level == .detailed else { return }
|
||||
logQueue.async {
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
debugInfoSync("🎯 [Decoded Response] [\(timestamp)] Type: \(type)")
|
||||
debugInfoSync("=====================================\n")
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
print("🎯 [Decoded Response] [\(timestamp)] Type: \(type)")
|
||||
print("=====================================\n")
|
||||
}
|
||||
|
||||
// MARK: - Helper Methods
|
||||
@@ -225,16 +293,20 @@ class APILogger {
|
||||
|
||||
// MARK: - Performance Logging
|
||||
static func logPerformanceWarning(duration: TimeInterval, threshold: TimeInterval = 5.0) {
|
||||
#if DEBUG
|
||||
guard logLevel != .none && duration > threshold else { return }
|
||||
#else
|
||||
#if !DEBUG
|
||||
return
|
||||
#else
|
||||
Task {
|
||||
let level = await Config.shared.get()
|
||||
guard level != .none && duration > threshold else { return }
|
||||
logQueue.async {
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
debugWarnSync("\n⚠️ [Performance Warning] [\(timestamp)] ============")
|
||||
debugWarnSync("🐌 Request took \(String(format: "%.3f", duration))s (threshold: \(threshold)s)")
|
||||
debugWarnSync("💡 建议:检查网络条件或优化 API 响应")
|
||||
debugWarnSync("================================================\n")
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
let timestamp = dateFormatter.string(from: Date())
|
||||
print("\n⚠️ [Performance Warning] [\(timestamp)] ============")
|
||||
print("🐌 Request took \(String(format: "%.3f", duration))s (threshold: \(threshold)s)")
|
||||
print("💡 建议:检查网络条件或优化 API 响应")
|
||||
print("================================================\n")
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
// MARK: - HTTP Method
|
||||
|
||||
@@ -205,8 +204,9 @@ struct BaseRequest: Codable {
|
||||
"\(key)=\(String(describing: filteredParams[key] ?? ""))"
|
||||
}.joined(separator: "&")
|
||||
|
||||
// 4. 添加密钥
|
||||
let keyString = "key=rpbs6us1m8r2j9g6u06ff2bo18orwaya"
|
||||
// 4. 添加密钥(从配置提供者获取)
|
||||
let key = SigningKeyProvider.signingKey()
|
||||
let keyString = "key=\(key)"
|
||||
let finalString = paramString.isEmpty ? keyString : "\(paramString)&\(keyString)"
|
||||
|
||||
// 5. 生成大写 MD5 签名
|
||||
@@ -217,9 +217,8 @@ struct BaseRequest: Codable {
|
||||
// MARK: - Network Type Detector
|
||||
struct NetworkTypeDetector {
|
||||
static func getCurrentNetworkType() -> Int {
|
||||
// WiFi = 2, 蜂窝网络 = 1
|
||||
// 这里是简化实现,实际应该检测网络状态
|
||||
return 2 // 默认蜂窝网络
|
||||
// WiFi = 2, 蜂窝网络 = 1, 其他/无网络 = 0
|
||||
return NetworkMonitor.shared.currentType
|
||||
}
|
||||
}
|
||||
|
||||
@@ -238,7 +237,6 @@ struct CarrierInfoManager {
|
||||
|
||||
// MARK: - User Info Manager (for Headers)
|
||||
struct UserInfoManager {
|
||||
@MainActor
|
||||
private static let keychain = KeychainManager.shared
|
||||
|
||||
// MARK: - Storage Keys
|
||||
@@ -287,7 +285,7 @@ struct UserInfoManager {
|
||||
// MARK: - User Info Management
|
||||
static func saveUserInfo(_ userInfo: UserInfo) async {
|
||||
do {
|
||||
try await keychain.store(userInfo, forKey: StorageKeys.userInfo)
|
||||
try keychain.store(userInfo, forKey: StorageKeys.userInfo)
|
||||
await cacheActor.setUserInfo(userInfo)
|
||||
debugInfoSync("💾 保存用户信息成功")
|
||||
} catch {
|
||||
@@ -302,7 +300,7 @@ struct UserInfoManager {
|
||||
}
|
||||
// 从 Keychain 读取
|
||||
do {
|
||||
let userInfo = try await keychain.retrieve(UserInfo.self, forKey: StorageKeys.userInfo)
|
||||
let userInfo = try keychain.retrieve(UserInfo.self, forKey: StorageKeys.userInfo)
|
||||
await cacheActor.setUserInfo(userInfo)
|
||||
return userInfo
|
||||
} catch {
|
||||
@@ -377,7 +375,7 @@ struct UserInfoManager {
|
||||
/// - Parameter accountModel: 要保存的账户模型
|
||||
static func saveAccountModel(_ accountModel: AccountModel) async {
|
||||
do {
|
||||
try await keychain.store(accountModel, forKey: StorageKeys.accountModel)
|
||||
try keychain.store(accountModel, forKey: StorageKeys.accountModel)
|
||||
await cacheActor.setAccountModel(accountModel)
|
||||
|
||||
// 同步更新 ticket 到内存
|
||||
@@ -400,7 +398,7 @@ struct UserInfoManager {
|
||||
}
|
||||
// 从 Keychain 读取
|
||||
do {
|
||||
let accountModel = try await keychain.retrieve(
|
||||
let accountModel = try keychain.retrieve(
|
||||
AccountModel.self,
|
||||
forKey: StorageKeys.accountModel
|
||||
)
|
||||
@@ -448,7 +446,7 @@ struct UserInfoManager {
|
||||
/// 清除 AccountModel
|
||||
static func clearAccountModel() async {
|
||||
do {
|
||||
try await keychain.delete(forKey: StorageKeys.accountModel)
|
||||
try keychain.delete(forKey: StorageKeys.accountModel)
|
||||
await cacheActor.clearAccountModel()
|
||||
debugInfoSync("🗑️ AccountModel 已清除")
|
||||
} catch {
|
||||
@@ -459,7 +457,7 @@ struct UserInfoManager {
|
||||
/// 清除用户信息
|
||||
static func clearUserInfo() async {
|
||||
do {
|
||||
try await keychain.delete(forKey: StorageKeys.userInfo)
|
||||
try keychain.delete(forKey: StorageKeys.userInfo)
|
||||
await cacheActor.clearUserInfo()
|
||||
debugInfoSync("🗑️ UserInfo 已清除")
|
||||
} catch {
|
||||
@@ -663,64 +661,11 @@ struct APIResponse<T: Codable>: Codable {
|
||||
|
||||
// MARK: - 腾讯云 COS Token 相关模型
|
||||
|
||||
/// 腾讯云 COS Token 请求模型
|
||||
struct TcTokenRequest: APIRequestProtocol {
|
||||
typealias Response = TcTokenResponse
|
||||
|
||||
let endpoint: String = APIEndpoint.tcToken.path
|
||||
let method: HTTPMethod = .GET
|
||||
let queryParameters: [String: String]? = nil
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
let timeout: TimeInterval = 30.0
|
||||
let includeBaseParameters: Bool = true
|
||||
let shouldShowLoading: Bool = false // 不显示 loading,避免影响用户体验
|
||||
let shouldShowError: Bool = false // 不显示错误,静默处理
|
||||
}
|
||||
// 注意:TcTokenRequest 和 TcTokenResponse 已迁移到 Utils/TCCos/Models/COSModels.swift
|
||||
// 请使用 COSModels.swift 中的版本
|
||||
|
||||
/// 腾讯云 COS Token 响应模型
|
||||
struct TcTokenResponse: Codable, Equatable {
|
||||
let code: Int
|
||||
let message: String
|
||||
let data: TcTokenData?
|
||||
let timestamp: Int64
|
||||
}
|
||||
|
||||
/// 腾讯云 COS Token 数据模型
|
||||
/// 包含完整的腾讯云 COS 配置信息
|
||||
struct TcTokenData: Codable, Equatable {
|
||||
let bucket: String // 存储桶名称
|
||||
let sessionToken: String // 临时会话令牌
|
||||
let region: String // 地域
|
||||
let customDomain: String // 自定义域名
|
||||
let accelerate: Bool // 是否启用加速
|
||||
let appId: String // 应用 ID
|
||||
let secretKey: String // 临时密钥
|
||||
let expireTime: Int64 // 过期时间戳
|
||||
let startTime: Int64 // 开始时间戳
|
||||
let secretId: String // 临时密钥 ID
|
||||
|
||||
/// 检查 Token 是否已过期
|
||||
var isExpired: Bool {
|
||||
let currentTime = Int64(Date().timeIntervalSince1970)
|
||||
return currentTime >= expireTime
|
||||
}
|
||||
|
||||
/// 获取过期时间
|
||||
var expirationDate: Date {
|
||||
return Date(timeIntervalSince1970: TimeInterval(expireTime))
|
||||
}
|
||||
|
||||
/// 获取开始时间
|
||||
var startDate: Date {
|
||||
return Date(timeIntervalSince1970: TimeInterval(startTime))
|
||||
}
|
||||
|
||||
/// 获取剩余有效时间(秒)
|
||||
var remainingTime: Int64 {
|
||||
let currentTime = Int64(Date().timeIntervalSince1970)
|
||||
return max(0, expireTime - currentTime)
|
||||
}
|
||||
}
|
||||
// 注意:TcTokenData 已迁移到 Utils/TCCos/Models/COSModels.swift
|
||||
// 请使用 COSModels.swift 中的 TcTokenData
|
||||
|
||||
// MARK: - User Info API Management
|
||||
extension UserInfoManager {
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
// MARK: - API Service Protocol
|
||||
|
||||
@@ -136,10 +135,7 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
requestBody = try JSONSerialization.data(withJSONObject: finalBody, options: [])
|
||||
urlRequest.httpBody = requestBody
|
||||
// urlRequest.setValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
|
||||
if let httpBody = urlRequest.httpBody,
|
||||
let bodyString = String(data: httpBody, encoding: .utf8) {
|
||||
debugInfoSync("HTTP Body: \(bodyString)")
|
||||
}
|
||||
// HTTP Body 的详细输出由 APILogger 统一处理(带脱敏)。这里不再重复输出。
|
||||
} catch {
|
||||
let encodingError = APIError.decodingError("请求体编码失败: \(error.localizedDescription)")
|
||||
await APILoadingManager.shared.setError(loadingId, errorMessage: encodingError.localizedDescription)
|
||||
@@ -148,8 +144,7 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
}
|
||||
|
||||
// 记录请求日志,传递完整的 headers 信息
|
||||
await APILogger
|
||||
.logRequest(request, url: url, body: requestBody, finalHeaders: headers)
|
||||
APILogger.logRequest(request, url: url, body: requestBody, finalHeaders: headers)
|
||||
|
||||
do {
|
||||
// 发起请求
|
||||
@@ -165,18 +160,16 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
|
||||
// 检查数据大小
|
||||
if data.count > APIConfiguration.maxDataSize {
|
||||
await APILogger
|
||||
.logError(APIError.resourceTooLarge, url: url, duration: duration)
|
||||
APILogger.logError(APIError.resourceTooLarge, url: url, duration: duration)
|
||||
await APILoadingManager.shared.setError(loadingId, errorMessage: APIError.resourceTooLarge.localizedDescription)
|
||||
throw APIError.resourceTooLarge
|
||||
}
|
||||
|
||||
// 记录响应日志
|
||||
await APILogger
|
||||
.logResponse(data: data, response: httpResponse, duration: duration)
|
||||
APILogger.logResponse(data: data, response: httpResponse, duration: duration)
|
||||
|
||||
// 性能警告
|
||||
await APILogger.logPerformanceWarning(duration: duration)
|
||||
APILogger.logPerformanceWarning(duration: duration)
|
||||
|
||||
// 检查 HTTP 状态码
|
||||
guard 200...299 ~= httpResponse.statusCode else {
|
||||
@@ -196,7 +189,7 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
do {
|
||||
let decoder = JSONDecoder()
|
||||
let decodedResponse = try decoder.decode(T.Response.self, from: data)
|
||||
await APILogger.logDecodedResponse(decodedResponse, type: T.Response.self)
|
||||
APILogger.logDecodedResponse(decodedResponse, type: T.Response.self)
|
||||
|
||||
// 请求成功,完成 loading
|
||||
await APILoadingManager.shared.finishLoading(loadingId)
|
||||
@@ -210,13 +203,13 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
|
||||
} catch let error as APIError {
|
||||
let duration = Date().timeIntervalSince(startTime)
|
||||
await APILogger.logError(error, url: url, duration: duration)
|
||||
APILogger.logError(error, url: url, duration: duration)
|
||||
await APILoadingManager.shared.setError(loadingId, errorMessage: error.localizedDescription)
|
||||
throw error
|
||||
} catch {
|
||||
let duration = Date().timeIntervalSince(startTime)
|
||||
let apiError = mapSystemError(error)
|
||||
await APILogger.logError(apiError, url: url, duration: duration)
|
||||
APILogger.logError(apiError, url: url, duration: duration)
|
||||
await APILoadingManager.shared.setError(loadingId, errorMessage: apiError.localizedDescription)
|
||||
throw apiError
|
||||
}
|
||||
@@ -300,6 +293,14 @@ struct LiveAPIService: APIServiceProtocol, Sendable {
|
||||
return error
|
||||
} else if let msg = json["msg"] as? String {
|
||||
return msg
|
||||
} else if let detail = json["detail"] as? String {
|
||||
return detail
|
||||
} else if let errorDescription = json["error_description"] as? String {
|
||||
return errorDescription
|
||||
} else if let errorDict = json["error"] as? [String: Any], let nestedMsg = errorDict["message"] as? String {
|
||||
return nestedMsg
|
||||
} else if let errors = json["errors"] as? [[String: Any]], let firstMsg = errors.first? ["message"] as? String {
|
||||
return firstMsg
|
||||
}
|
||||
|
||||
return nil
|
||||
@@ -349,7 +350,9 @@ actor MockAPIServiceActor: APIServiceProtocol, Sendable {
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - TCA Dependency Integration
|
||||
// MARK: - TCA Dependency Integration (optional)
|
||||
#if canImport(ComposableArchitecture)
|
||||
import ComposableArchitecture
|
||||
private enum APIServiceKey: DependencyKey {
|
||||
static let liveValue: any APIServiceProtocol & Sendable = LiveAPIService()
|
||||
static let testValue: any APIServiceProtocol & Sendable = MockAPIServiceActor()
|
||||
@@ -361,6 +364,7 @@ extension DependencyValues {
|
||||
set { self[APIServiceKey.self] = newValue }
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
// MARK: - BaseRequest Dictionary Conversion
|
||||
extension BaseRequest {
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
// MARK: - 响应数据模型
|
||||
|
||||
@@ -18,7 +17,7 @@ struct MomentsListData: Codable, Equatable, Sendable {
|
||||
}
|
||||
|
||||
/// 动态信息结构
|
||||
public struct MomentsInfo: Codable, Equatable, Sendable {
|
||||
public struct MomentsInfo: Codable, Equatable, Sendable, Identifiable {
|
||||
let dynamicId: Int
|
||||
let uid: Int
|
||||
let nick: String
|
||||
@@ -52,6 +51,7 @@ public struct MomentsInfo: Codable, Equatable, Sendable {
|
||||
let isCustomWord: Bool?
|
||||
let labelList: [String]?
|
||||
// 计算属性
|
||||
public var id: Int { dynamicId } // Identifiable 协议要求
|
||||
var isSquareTop: Bool { (squareTop ?? 0) != 0 }
|
||||
var isTopicTop: Bool { (topicTop ?? 0) != 0 }
|
||||
var formattedPublishTime: Date {
|
||||
@@ -241,12 +241,68 @@ struct PublishFeedData: Codable, Equatable {
|
||||
|
||||
// MARK: - 我的动态 API 请求
|
||||
|
||||
/// 我的动态信息结构 - 专门用于 /dynamic/getMyDynamic 接口
|
||||
struct MyMomentInfo: Codable, Equatable, Sendable {
|
||||
// 服务器可能返回的完整字段(均用可选兼容不同版本)
|
||||
let dynamicId: Int?
|
||||
let uid: Int
|
||||
let nick: String?
|
||||
let avatar: String?
|
||||
let type: Int
|
||||
let content: String
|
||||
let likeCount: Int?
|
||||
let isLike: Bool?
|
||||
let commentCount: Int?
|
||||
let publishTime: Int64
|
||||
let worldId: Int?
|
||||
let status: Int?
|
||||
let playCount: Int?
|
||||
let dynamicResList: [MomentsPicture]? // 资源列表(图片/视频)
|
||||
|
||||
// 转换为 MomentsInfo 的辅助方法
|
||||
func toMomentsInfo() -> MomentsInfo {
|
||||
return MomentsInfo(
|
||||
dynamicId: dynamicId ?? 0,
|
||||
uid: uid,
|
||||
nick: nick ?? "",
|
||||
avatar: avatar ?? "",
|
||||
type: type,
|
||||
content: content,
|
||||
likeCount: likeCount ?? 0,
|
||||
isLike: isLike ?? false,
|
||||
commentCount: commentCount ?? 0,
|
||||
// 注意:UI 的 formatDisplayTime 期望毫秒,这里不做 /1000 转换
|
||||
publishTime: Int(publishTime),
|
||||
worldId: worldId ?? 0,
|
||||
status: status ?? 1,
|
||||
playCount: playCount,
|
||||
dynamicResList: dynamicResList,
|
||||
gender: nil,
|
||||
squareTop: nil,
|
||||
topicTop: nil,
|
||||
newUser: nil,
|
||||
defUser: nil,
|
||||
scene: nil,
|
||||
userVipInfoVO: nil,
|
||||
headwearPic: nil,
|
||||
headwearEffect: nil,
|
||||
headwearType: nil,
|
||||
headwearName: nil,
|
||||
headwearId: nil,
|
||||
experLevelPic: nil,
|
||||
charmLevelPic: nil,
|
||||
isCustomWord: nil,
|
||||
labelList: nil
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/// 我的动态响应结构
|
||||
struct MyMomentsResponse: Codable, Equatable, Sendable {
|
||||
let code: Int
|
||||
let message: String
|
||||
let data: [MomentsInfo]?
|
||||
let timestamp: Int?
|
||||
let data: [MyMomentInfo]?
|
||||
let timestamp: Int64?
|
||||
}
|
||||
|
||||
struct GetMyDynamicRequest: APIRequestProtocol {
|
||||
@@ -281,3 +337,101 @@ struct GetMyDynamicRequest: APIRequestProtocol {
|
||||
var shouldShowLoading: Bool { true }
|
||||
var shouldShowError: Bool { true }
|
||||
}
|
||||
|
||||
// MARK: - 动态点赞 API 请求与响应
|
||||
|
||||
/// 动态点赞响应结构
|
||||
struct LikeDynamicResponse: Codable, Equatable, Sendable {
|
||||
let code: Int
|
||||
let message: String
|
||||
let data: LikeDynamicData?
|
||||
let timestamp: Int?
|
||||
}
|
||||
|
||||
/// 动态点赞返回数据
|
||||
struct LikeDynamicData: Codable, Equatable, Sendable {
|
||||
let success: Bool?
|
||||
let likeCount: Int?
|
||||
}
|
||||
|
||||
/// 动态点赞请求
|
||||
struct LikeDynamicRequest: APIRequestProtocol {
|
||||
typealias Response = LikeDynamicResponse
|
||||
|
||||
let endpoint: String = APIEndpoint.dynamicLike.path
|
||||
let method: HTTPMethod = .POST
|
||||
|
||||
let dynamicId: Int
|
||||
let uid: Int
|
||||
let status: Int // 0: 取消点赞, 1: 点赞
|
||||
let likedUid: Int
|
||||
let worldId: Int
|
||||
|
||||
init(dynamicId: Int, uid: Int, status: Int, likedUid: Int, worldId: Int) {
|
||||
self.dynamicId = dynamicId
|
||||
self.uid = uid
|
||||
self.status = status
|
||||
self.likedUid = likedUid
|
||||
self.worldId = worldId
|
||||
}
|
||||
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
|
||||
var queryParameters: [String: String]? {
|
||||
return [
|
||||
"dynamicId": String(dynamicId),
|
||||
"uid": String(uid),
|
||||
"status": String(status),
|
||||
"likedUid": String(likedUid),
|
||||
"worldId": String(worldId)
|
||||
]
|
||||
}
|
||||
|
||||
var includeBaseParameters: Bool { true }
|
||||
var shouldShowLoading: Bool { true }
|
||||
var shouldShowError: Bool { true }
|
||||
}
|
||||
|
||||
// MARK: - 删除动态 API 请求与响应
|
||||
|
||||
/// 删除动态响应结构
|
||||
struct DeleteDynamicResponse: Codable, Equatable, Sendable {
|
||||
let code: Int
|
||||
let message: String
|
||||
let data: DeleteDynamicData?
|
||||
let timestamp: Int?
|
||||
}
|
||||
|
||||
/// 删除动态返回数据
|
||||
struct DeleteDynamicData: Codable, Equatable, Sendable {
|
||||
let success: Bool?
|
||||
}
|
||||
|
||||
/// 删除动态请求
|
||||
struct DeleteDynamicRequest: APIRequestProtocol {
|
||||
typealias Response = DeleteDynamicResponse
|
||||
|
||||
let endpoint: String = APIEndpoint.deleteDynamic.path
|
||||
let method: HTTPMethod = .POST
|
||||
|
||||
let dynamicId: Int
|
||||
let uid: Int
|
||||
|
||||
init(dynamicId: Int, uid: Int) {
|
||||
self.dynamicId = dynamicId
|
||||
self.uid = uid
|
||||
}
|
||||
|
||||
var queryParameters: [String: String]? { nil }
|
||||
|
||||
var bodyParameters: [String: Any]? {
|
||||
return [
|
||||
"dynamicId": dynamicId,
|
||||
"uid": uid
|
||||
]
|
||||
}
|
||||
|
||||
var includeBaseParameters: Bool { true }
|
||||
var shouldShowLoading: Bool { true }
|
||||
var shouldShowError: Bool { true }
|
||||
}
|
||||
|
||||
@@ -77,10 +77,29 @@ struct IDLoginAPIRequest: APIRequestProtocol {
|
||||
let endpoint = APIEndpoint.login.path // 使用枚举定义的登录端点
|
||||
let method: HTTPMethod = .POST
|
||||
let includeBaseParameters = true
|
||||
let queryParameters: [String: String]?
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
let timeout: TimeInterval = 30.0
|
||||
|
||||
// MARK: - Private Properties
|
||||
private let phone: String
|
||||
private let password: String
|
||||
private let clientSecret: String
|
||||
private let version: String
|
||||
private let clientId: String
|
||||
private let grantType: String
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var queryParameters: [String: String]? {
|
||||
return [
|
||||
"phone": phone,
|
||||
"password": password,
|
||||
"client_secret": clientSecret,
|
||||
"version": version,
|
||||
"client_id": clientId,
|
||||
"grant_type": grantType
|
||||
]
|
||||
}
|
||||
|
||||
/// 初始化ID登录请求
|
||||
/// - Parameters:
|
||||
/// - phone: DES加密后的用户ID/手机号
|
||||
@@ -90,14 +109,12 @@ struct IDLoginAPIRequest: APIRequestProtocol {
|
||||
/// - clientId: 客户端ID,固定为"erban-client"
|
||||
/// - grantType: 授权类型,固定为"password"
|
||||
init(phone: String, password: String, clientSecret: String = "uyzjdhds", version: String = "1", clientId: String = "erban-client", grantType: String = "password") {
|
||||
self.queryParameters = [
|
||||
"phone": phone,
|
||||
"password": password,
|
||||
"client_secret": clientSecret,
|
||||
"version": version,
|
||||
"client_id": clientId,
|
||||
"grant_type": grantType
|
||||
];
|
||||
self.phone = phone
|
||||
self.password = password
|
||||
self.clientSecret = clientSecret
|
||||
self.version = version
|
||||
self.clientId = clientId
|
||||
self.grantType = grantType
|
||||
}
|
||||
}
|
||||
|
||||
@@ -375,7 +392,7 @@ struct LoginHelper {
|
||||
debugInfoSync(" 加密后密码: \(encryptedPassword)")
|
||||
|
||||
return IDLoginAPIRequest(
|
||||
phone: userID,
|
||||
phone: encryptedID,
|
||||
password: encryptedPassword
|
||||
)
|
||||
}
|
||||
@@ -527,10 +544,29 @@ struct EmailLoginRequest: APIRequestProtocol {
|
||||
let endpoint = APIEndpoint.login.path
|
||||
let method: HTTPMethod = .POST
|
||||
let includeBaseParameters = true
|
||||
let queryParameters: [String: String]?
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
let timeout: TimeInterval = 30.0
|
||||
|
||||
// MARK: - Private Properties
|
||||
private let email: String
|
||||
private let code: String
|
||||
private let clientSecret: String
|
||||
private let version: String
|
||||
private let clientId: String
|
||||
private let grantType: String
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var queryParameters: [String: String]? {
|
||||
return [
|
||||
"email": email,
|
||||
"code": code,
|
||||
"client_secret": clientSecret,
|
||||
"version": version,
|
||||
"client_id": clientId,
|
||||
"grant_type": grantType
|
||||
]
|
||||
}
|
||||
|
||||
/// 初始化邮箱验证码登录请求
|
||||
/// - Parameters:
|
||||
/// - email: DES加密后的邮箱地址
|
||||
@@ -540,14 +576,12 @@ struct EmailLoginRequest: APIRequestProtocol {
|
||||
/// - clientId: 客户端ID,固定为"erban-client"
|
||||
/// - grantType: 授权类型,固定为"email"
|
||||
init(email: String, code: String, clientSecret: String = "uyzjdhds", version: String = "1", clientId: String = "erban-client", grantType: String = "email") {
|
||||
self.queryParameters = [
|
||||
"email": email,
|
||||
"code": code,
|
||||
"client_secret": clientSecret,
|
||||
"version": version,
|
||||
"client_id": clientId,
|
||||
"grant_type": grantType
|
||||
]
|
||||
self.email = email
|
||||
self.code = code
|
||||
self.clientSecret = clientSecret
|
||||
self.version = version
|
||||
self.clientId = clientId
|
||||
self.grantType = grantType
|
||||
}
|
||||
}
|
||||
|
||||
@@ -603,18 +637,25 @@ struct GetUserInfoRequest: APIRequestProtocol {
|
||||
let endpoint = APIEndpoint.getUserInfo.path
|
||||
let method: HTTPMethod = .GET
|
||||
let includeBaseParameters = true
|
||||
let queryParameters: [String: String]?
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
let timeout: TimeInterval = 30.0
|
||||
let shouldShowLoading: Bool = false // 不显示loading,避免影响用户体验
|
||||
let shouldShowError: Bool = false // 不显示错误,静默处理
|
||||
|
||||
// MARK: - Private Properties
|
||||
private let uid: String
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var queryParameters: [String: String]? {
|
||||
return [
|
||||
"uid": uid
|
||||
]
|
||||
}
|
||||
|
||||
/// 初始化获取用户信息请求
|
||||
/// - Parameter uid: 要查询的用户ID
|
||||
init(uid: String) {
|
||||
self.queryParameters = [
|
||||
"uid": uid
|
||||
]
|
||||
self.uid = uid
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -2,12 +2,16 @@ import UIKit
|
||||
//import NIMSDK
|
||||
|
||||
class AppDelegate: UIResponder, UIApplicationDelegate {
|
||||
private func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) async -> Bool {
|
||||
|
||||
// 预加载用户信息缓存
|
||||
await UserInfoManager.preloadCache()
|
||||
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
|
||||
debugInfoSync("🚀 UIApplication didFinishLaunching")
|
||||
|
||||
// NIMConfigurationManager.setupNimSDK()
|
||||
// 异步预加载用户信息缓存与初始化逻辑(不阻塞启动)
|
||||
Task { @MainActor in
|
||||
await UserInfoManager.preloadCache()
|
||||
// 如需集成 IM/其他 SDK,在此处异步初始化,避免阻塞:
|
||||
// NIMConfigurationManager.setupNimSDK()
|
||||
debugInfoSync("✅ App 启动预热完成")
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
6
yana/Assets.xcassets/Common/Contents.json
Normal file
6
yana/Assets.xcassets/Common/Contents.json
Normal file
@@ -0,0 +1,6 @@
|
||||
{
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
||||
21
yana/Assets.xcassets/Common/icon_copy.imageset/Contents.json
vendored
Normal file
21
yana/Assets.xcassets/Common/icon_copy.imageset/Contents.json
vendored
Normal file
@@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"filename" : "复制@3x.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
||||
BIN
yana/Assets.xcassets/Common/icon_copy.imageset/复制@3x.png
vendored
Normal file
BIN
yana/Assets.xcassets/Common/icon_copy.imageset/复制@3x.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 646 B |
21
yana/Assets.xcassets/Home/setting icon.imageset/Contents.json
vendored
Normal file
21
yana/Assets.xcassets/Home/setting icon.imageset/Contents.json
vendored
Normal file
@@ -0,0 +1,21 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"filename" : "切图 12@3x.png",
|
||||
"idiom" : "universal",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
||||
BIN
yana/Assets.xcassets/Home/setting icon.imageset/切图 12@3x.png
vendored
Normal file
BIN
yana/Assets.xcassets/Home/setting icon.imageset/切图 12@3x.png
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.8 KiB |
@@ -1,10 +1,10 @@
|
||||
enum Environment {
|
||||
enum AppEnvironment {
|
||||
case development
|
||||
case production
|
||||
}
|
||||
|
||||
struct AppConfig {
|
||||
static let current: Environment = {
|
||||
static let current: AppEnvironment = {
|
||||
#if DEBUG
|
||||
return .development
|
||||
#else
|
||||
@@ -15,8 +15,7 @@ struct AppConfig {
|
||||
static var baseURL: String {
|
||||
switch current {
|
||||
case .development:
|
||||
// return "http://192.168.10.211:8080"
|
||||
return "http://beta.api.molistar.xyz"
|
||||
return "http://beta.api.pekolive.com"
|
||||
case .production:
|
||||
return "https://api.epartylive.com"
|
||||
}
|
||||
|
||||
@@ -170,7 +170,14 @@ struct ContentView: View {
|
||||
let store: StoreOf<LoginFeature>
|
||||
let initStore: StoreOf<InitFeature>
|
||||
let configStore: StoreOf<ConfigFeature>
|
||||
@State private var selectedLogLevel: APILogger.LogLevel = APILogger.logLevel
|
||||
@State private var selectedLogLevel: APILogger.LogLevel = {
|
||||
// 以编译期默认值初始化(与 APILogger.Config 一致)
|
||||
#if DEBUG
|
||||
return .detailed
|
||||
#else
|
||||
return .none
|
||||
#endif
|
||||
}()
|
||||
@State private var selectedTab = 0
|
||||
|
||||
var body: some View {
|
||||
@@ -187,8 +194,8 @@ struct ContentView: View {
|
||||
}
|
||||
.tag(1)
|
||||
}
|
||||
.onChange(of: selectedLogLevel) { newValue in
|
||||
APILogger.logLevel = newValue
|
||||
.onChange(of: selectedLogLevel) { _, selectedLogLevel in
|
||||
Task { await APILogger.Config.shared.set(selectedLogLevel) }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
import SwiftUI
|
||||
import PhotosUI
|
||||
|
||||
@Reducer
|
||||
struct AppSettingFeature {
|
||||
@@ -14,6 +16,7 @@ struct AppSettingFeature {
|
||||
// WebView 导航状态
|
||||
var showUserAgreement: Bool = false
|
||||
var showPrivacyPolicy: Bool = false
|
||||
var showDeactivateAccount: Bool = false
|
||||
|
||||
// 头像/昵称修改相关
|
||||
var isUploadingAvatar: Bool = false
|
||||
@@ -23,14 +26,29 @@ struct AppSettingFeature {
|
||||
var isUpdatingUser: Bool = false
|
||||
var updateUserError: String? = nil
|
||||
|
||||
// 新增:带userInfo、avatarURL、nickname的init
|
||||
// 默认初始化器
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
|
||||
// 带userInfo、avatarURL、nickname的init
|
||||
init(nickname: String = "", avatarURL: String? = nil, userInfo: UserInfo? = nil) {
|
||||
self.nickname = nickname
|
||||
self.avatarURL = avatarURL
|
||||
self.userInfo = userInfo
|
||||
}
|
||||
// 新增:TCA驱动图片选择弹窗
|
||||
var showImagePicker: Bool = false
|
||||
|
||||
// 图片源选择 ActionSheet
|
||||
var showImageSourceActionSheet: Bool = false
|
||||
|
||||
// 新增:直接管理相机和相册选择
|
||||
var showCamera: Bool = false
|
||||
var showPhotoPicker: Bool = false
|
||||
var selectedPhotoItems: [PhotosPickerItem] = []
|
||||
|
||||
// 弹窗状态
|
||||
var showLogoutConfirmation: Bool = false
|
||||
var showAboutUs: Bool = false
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
@@ -49,10 +67,12 @@ struct AppSettingFeature {
|
||||
case clearCacheTapped
|
||||
case checkUpdatesTapped
|
||||
case aboutUsTapped
|
||||
case deactivateAccountTapped
|
||||
|
||||
// WebView 关闭
|
||||
case userAgreementDismissed
|
||||
case privacyPolicyDismissed
|
||||
case deactivateAccountDismissed
|
||||
|
||||
// 头像/昵称修改
|
||||
case avatarTapped
|
||||
@@ -63,8 +83,21 @@ struct AppSettingFeature {
|
||||
case nicknameInputChanged(String)
|
||||
case nicknameEditAlert(Bool)
|
||||
case testPushTapped
|
||||
// 新增:TCA驱动图片选择弹窗
|
||||
case setShowImagePicker(Bool)
|
||||
|
||||
// 图片源选择
|
||||
case setShowImageSourceActionSheet(Bool)
|
||||
case selectImageSource(AppImageSource)
|
||||
|
||||
// 新增:直接处理相机和相册
|
||||
case setShowCamera(Bool)
|
||||
case setShowPhotoPicker(Bool)
|
||||
case cameraImagePicked(UIImage?)
|
||||
case photoPickerItemsChanged([PhotosPickerItem])
|
||||
|
||||
// 弹窗相关
|
||||
case showLogoutConfirmation(Bool)
|
||||
case showAboutUs(Bool)
|
||||
case logoutConfirmed
|
||||
}
|
||||
|
||||
@Dependency(\.apiService) var apiService
|
||||
@@ -79,6 +112,11 @@ struct AppSettingFeature {
|
||||
return .none
|
||||
|
||||
case .logoutTapped:
|
||||
// 显示登出确认弹窗
|
||||
state.showLogoutConfirmation = true
|
||||
return .none
|
||||
|
||||
case .logoutConfirmed:
|
||||
// 清理所有认证信息,并向上层发送登出事件
|
||||
return .run { send in
|
||||
await UserInfoManager.clearAllAuthenticationData()
|
||||
@@ -140,7 +178,11 @@ struct AppSettingFeature {
|
||||
return .none
|
||||
|
||||
case .aboutUsTapped:
|
||||
// 预留关于我们逻辑
|
||||
state.showAboutUs = true
|
||||
return .none
|
||||
|
||||
case .deactivateAccountTapped:
|
||||
state.showDeactivateAccount = true
|
||||
return .none
|
||||
|
||||
case .userAgreementDismissed:
|
||||
@@ -151,6 +193,10 @@ struct AppSettingFeature {
|
||||
state.showPrivacyPolicy = false
|
||||
return .none
|
||||
|
||||
case .deactivateAccountDismissed:
|
||||
state.showDeactivateAccount = false
|
||||
return .none
|
||||
|
||||
case .avatarTapped:
|
||||
// 触发头像选择器
|
||||
return .none
|
||||
@@ -170,8 +216,6 @@ struct AppSettingFeature {
|
||||
}
|
||||
}
|
||||
case let .avatarUploadResult(.success(url)):
|
||||
state.isUploadingAvatar = false
|
||||
// 调用 updateUser API,仅传 avatar
|
||||
state.isUpdatingUser = true
|
||||
state.updateUserError = nil
|
||||
guard let userInfo = state.userInfo else { return .none }
|
||||
@@ -235,8 +279,63 @@ struct AppSettingFeature {
|
||||
return .none
|
||||
case .testPushTapped:
|
||||
return .none
|
||||
case .setShowImagePicker(let show):
|
||||
state.showImagePicker = show
|
||||
|
||||
// 图片源选择处理
|
||||
case .setShowImageSourceActionSheet(let show):
|
||||
state.showImageSourceActionSheet = show
|
||||
return .none
|
||||
|
||||
case .selectImageSource(let source):
|
||||
state.showImageSourceActionSheet = false
|
||||
switch source {
|
||||
case .camera:
|
||||
state.showCamera = true
|
||||
case .photoLibrary:
|
||||
state.showPhotoPicker = true
|
||||
}
|
||||
return .none
|
||||
|
||||
// 相机和相册处理
|
||||
case .setShowCamera(let show):
|
||||
state.showCamera = show
|
||||
return .none
|
||||
|
||||
case .setShowPhotoPicker(let show):
|
||||
state.showPhotoPicker = show
|
||||
return .none
|
||||
|
||||
case .cameraImagePicked(let image):
|
||||
state.showCamera = false
|
||||
if let image = image,
|
||||
let imageData = image.jpegData(compressionQuality: 0.8) {
|
||||
return .send(.avatarSelected(imageData))
|
||||
}
|
||||
return .none
|
||||
|
||||
case .photoPickerItemsChanged(let items):
|
||||
state.selectedPhotoItems = items
|
||||
if !items.isEmpty {
|
||||
state.showPhotoPicker = false
|
||||
// 处理选中的图片
|
||||
return .run { send in
|
||||
for item in items {
|
||||
if let data = try? await item.loadTransferable(type: Data.self),
|
||||
let image = UIImage(data: data),
|
||||
let imageData = image.jpegData(compressionQuality: 0.8) {
|
||||
await send(.avatarSelected(imageData))
|
||||
break // 只处理第一张图片
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return .none
|
||||
|
||||
case .showLogoutConfirmation(let show):
|
||||
state.showLogoutConfirmation = show
|
||||
return .none
|
||||
|
||||
case .showAboutUs(let show):
|
||||
state.showAboutUs = show
|
||||
return .none
|
||||
}
|
||||
}
|
||||
|
||||
@@ -40,6 +40,10 @@ struct ConfigFeature {
|
||||
var configData: ConfigData?
|
||||
var errorMessage: String?
|
||||
var lastUpdated: Date?
|
||||
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
|
||||
@@ -5,157 +5,205 @@ struct ConfigView: View {
|
||||
let store: StoreOf<ConfigFeature>
|
||||
|
||||
var body: some View {
|
||||
WithPerceptionTracking {
|
||||
NavigationView {
|
||||
VStack(spacing: 20) {
|
||||
// 标题
|
||||
Text("API 配置测试")
|
||||
.font(.largeTitle)
|
||||
.fontWeight(.bold)
|
||||
.padding(.top)
|
||||
|
||||
// 状态显示
|
||||
Group {
|
||||
if store.isLoading {
|
||||
VStack {
|
||||
ProgressView()
|
||||
.scaleEffect(1.5)
|
||||
Text("正在加载配置...")
|
||||
.font(.headline)
|
||||
.foregroundColor(.secondary)
|
||||
.padding(.top, 8)
|
||||
}
|
||||
.frame(height: 100)
|
||||
} else if let errorMessage = store.errorMessage {
|
||||
VStack {
|
||||
Image(systemName: "exclamationmark.triangle.fill")
|
||||
.font(.system(size: 40))
|
||||
.foregroundColor(.red)
|
||||
|
||||
Text("错误")
|
||||
.font(.headline)
|
||||
.fontWeight(.semibold)
|
||||
|
||||
Text(errorMessage)
|
||||
.font(.body)
|
||||
.foregroundColor(.secondary)
|
||||
.multilineTextAlignment(.center)
|
||||
.padding(.horizontal)
|
||||
|
||||
Button("清除错误") {
|
||||
store.send(.clearError)
|
||||
}
|
||||
.buttonStyle(.borderedProminent)
|
||||
.padding(.top)
|
||||
}
|
||||
.frame(maxHeight: .infinity)
|
||||
} else if let configData = store.configData {
|
||||
// 配置数据显示
|
||||
ScrollView {
|
||||
VStack(alignment: .leading, spacing: 16) {
|
||||
|
||||
if let version = configData.version {
|
||||
InfoRow(title: "版本", value: version)
|
||||
}
|
||||
|
||||
if let features = configData.features, !features.isEmpty {
|
||||
VStack(alignment: .leading, spacing: 8) {
|
||||
Text("功能列表")
|
||||
.font(.headline)
|
||||
.fontWeight(.semibold)
|
||||
|
||||
ForEach(features, id: \.self) { feature in
|
||||
HStack {
|
||||
Circle()
|
||||
.fill(Color.green)
|
||||
.frame(width: 6, height: 6)
|
||||
Text(feature)
|
||||
.font(.body)
|
||||
}
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
.background(Color(.systemGray6))
|
||||
.cornerRadius(12)
|
||||
}
|
||||
|
||||
if let settings = configData.settings {
|
||||
VStack(alignment: .leading, spacing: 8) {
|
||||
Text("设置")
|
||||
.font(.headline)
|
||||
.fontWeight(.semibold)
|
||||
|
||||
if let enableDebug = settings.enableDebug {
|
||||
InfoRow(title: "调试模式", value: enableDebug ? "启用" : "禁用")
|
||||
}
|
||||
|
||||
if let apiTimeout = settings.apiTimeout {
|
||||
InfoRow(title: "API 超时", value: "\(apiTimeout)秒")
|
||||
}
|
||||
|
||||
if let maxRetries = settings.maxRetries {
|
||||
InfoRow(title: "最大重试次数", value: "\(maxRetries)")
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
.background(Color(.systemGray6))
|
||||
.cornerRadius(12)
|
||||
}
|
||||
|
||||
if let lastUpdated = store.lastUpdated {
|
||||
Text("最后更新: \(lastUpdated, style: .time)")
|
||||
.font(.caption)
|
||||
.foregroundColor(.secondary)
|
||||
.frame(maxWidth: .infinity, alignment: .center)
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
}
|
||||
} else {
|
||||
VStack {
|
||||
Image(systemName: "gear")
|
||||
.font(.system(size: 40))
|
||||
.foregroundColor(.secondary)
|
||||
|
||||
Text("点击下方按钮加载配置")
|
||||
.font(.headline)
|
||||
.foregroundColor(.secondary)
|
||||
}
|
||||
.frame(maxHeight: .infinity)
|
||||
}
|
||||
NavigationView {
|
||||
VStack(spacing: 20) {
|
||||
Text(LocalizedString("config.api_test", comment: ""))
|
||||
.font(.largeTitle)
|
||||
.fontWeight(.bold)
|
||||
.padding(.top)
|
||||
|
||||
// 状态显示
|
||||
Group {
|
||||
if store.isLoading {
|
||||
LoadingView()
|
||||
} else if store.errorMessage != nil {
|
||||
ConfigErrorView(store: store)
|
||||
} else if let configData = store.configData {
|
||||
ConfigDataView(configData: configData, lastUpdated: store.lastUpdated)
|
||||
} else {
|
||||
// EmptyStateView()
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
// 操作按钮
|
||||
VStack(spacing: 12) {
|
||||
Button(action: {
|
||||
store.send(.loadConfig)
|
||||
}) {
|
||||
HStack {
|
||||
if store.isLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
} else {
|
||||
Image(systemName: "arrow.clockwise")
|
||||
}
|
||||
Text(store.isLoading ? "加载中..." : "加载配置")
|
||||
}
|
||||
}
|
||||
.buttonStyle(.borderedProminent)
|
||||
.disabled(store.isLoading)
|
||||
.frame(maxWidth: .infinity)
|
||||
.frame(height: 50)
|
||||
|
||||
Text("使用新的 TCA API 组件")
|
||||
.font(.caption)
|
||||
.foregroundColor(.secondary)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
// 操作按钮
|
||||
ActionButtonsView(store: store)
|
||||
|
||||
}
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Loading View
|
||||
struct LoadingView: View {
|
||||
var body: some View {
|
||||
VStack {
|
||||
ProgressView()
|
||||
.scaleEffect(1.2)
|
||||
Text(LocalizedString("config.loading", comment: ""))
|
||||
.font(.caption)
|
||||
.foregroundColor(.secondary)
|
||||
.padding(.top, 8)
|
||||
}
|
||||
.frame(height: 100)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Error View
|
||||
struct ConfigErrorView: View {
|
||||
let store: StoreOf<ConfigFeature>
|
||||
|
||||
var body: some View {
|
||||
VStack(spacing: 16) {
|
||||
Image(systemName: "exclamationmark.triangle.fill")
|
||||
.font(.system(size: 40))
|
||||
.foregroundColor(.yellow)
|
||||
Text(LocalizedString("config.error", comment: ""))
|
||||
.foregroundColor(.red)
|
||||
Button(LocalizedString("config.clear_error", comment: "")) {
|
||||
store.send(.clearError)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Config Data View
|
||||
struct ConfigDataView: View {
|
||||
let configData: ConfigData
|
||||
let lastUpdated: Date?
|
||||
|
||||
var body: some View {
|
||||
ScrollView {
|
||||
VStack(alignment: .leading, spacing: 16) {
|
||||
if let version = configData.version {
|
||||
InfoRow(title: LocalizedString("config.version", comment: ""), value: version)
|
||||
}
|
||||
|
||||
if let features = configData.features, !features.isEmpty {
|
||||
FeaturesSection(features: features)
|
||||
}
|
||||
|
||||
if let settings = configData.settings {
|
||||
SettingsSection(settings: settings)
|
||||
}
|
||||
|
||||
if let lastUpdated = lastUpdated {
|
||||
Text(String(format: LocalizedString("config.last_updated", comment: ""), {
|
||||
let formatter = DateFormatter()
|
||||
formatter.dateStyle = .medium
|
||||
formatter.timeStyle = .short
|
||||
return formatter.string(from: lastUpdated)
|
||||
}()))
|
||||
.font(.caption)
|
||||
.foregroundColor(.secondary)
|
||||
.frame(maxWidth: .infinity, alignment: .center)
|
||||
}
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.padding()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Features Section
|
||||
struct FeaturesSection: View {
|
||||
let features: [String]
|
||||
|
||||
var body: some View {
|
||||
VStack(alignment: .leading, spacing: 8) {
|
||||
Text(LocalizedString("config.feature_list", comment: ""))
|
||||
.font(.headline)
|
||||
.fontWeight(.semibold)
|
||||
|
||||
ForEach(features, id: \.self) { feature in
|
||||
HStack {
|
||||
Circle()
|
||||
.fill(Color.green)
|
||||
.frame(width: 6, height: 6)
|
||||
Text(feature)
|
||||
.font(.body)
|
||||
}
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
.background(Color(.systemGray6))
|
||||
.cornerRadius(12)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Settings Section
|
||||
struct SettingsSection: View {
|
||||
let settings: ConfigSettings
|
||||
|
||||
var body: some View {
|
||||
VStack(alignment: .leading, spacing: 8) {
|
||||
Text(LocalizedString("config.settings", comment: ""))
|
||||
.font(.headline)
|
||||
.fontWeight(.semibold)
|
||||
|
||||
if let enableDebug = settings.enableDebug {
|
||||
InfoRow(title: LocalizedString("config.debug_mode", comment: ""), value: enableDebug ? "启用" : "禁用")
|
||||
}
|
||||
|
||||
if let apiTimeout = settings.apiTimeout {
|
||||
InfoRow(title: LocalizedString("config.api_timeout", comment: ""), value: "\(apiTimeout)秒")
|
||||
}
|
||||
|
||||
if let maxRetries = settings.maxRetries {
|
||||
InfoRow(title: LocalizedString("config.max_retries", comment: ""), value: "\(maxRetries)")
|
||||
}
|
||||
}
|
||||
.padding()
|
||||
.background(Color(.systemGray6))
|
||||
.cornerRadius(12)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Empty State View
|
||||
//struct EmptyStateView: View {
|
||||
// var body: some View {
|
||||
// VStack(spacing: 16) {
|
||||
// Image(systemName: "arrow.down.circle")
|
||||
// .font(.system(size: 40))
|
||||
// .foregroundColor(.blue)
|
||||
// Text(LocalizedString("config.click_to_load", comment: ""))
|
||||
// .font(.body)
|
||||
// .multilineTextAlignment(.center)
|
||||
// .foregroundColor(.secondary)
|
||||
// }
|
||||
// .frame(maxHeight: .infinity)
|
||||
// }
|
||||
//}
|
||||
|
||||
// MARK: - Action Buttons View
|
||||
struct ActionButtonsView: View {
|
||||
let store: StoreOf<ConfigFeature>
|
||||
|
||||
var body: some View {
|
||||
VStack(spacing: 12) {
|
||||
Button(action: {
|
||||
store.send(.loadConfig)
|
||||
}) {
|
||||
HStack {
|
||||
if store.isLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
} else {
|
||||
Image(systemName: "arrow.clockwise")
|
||||
}
|
||||
Text(store.isLoading ? "加载中..." : "加载配置")
|
||||
}
|
||||
}
|
||||
.buttonStyle(.borderedProminent)
|
||||
.disabled(store.isLoading)
|
||||
.frame(maxWidth: .infinity)
|
||||
.frame(height: 50)
|
||||
|
||||
Text(LocalizedString("config.use_new_tca", comment: ""))
|
||||
.font(.caption)
|
||||
.foregroundColor(.secondary)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -181,10 +229,10 @@ struct InfoRow: View {
|
||||
}
|
||||
|
||||
// MARK: - Preview
|
||||
#Preview {
|
||||
ConfigView(
|
||||
store: Store(initialState: ConfigFeature.State()) {
|
||||
ConfigFeature()
|
||||
}
|
||||
)
|
||||
}
|
||||
//#Preview {
|
||||
// ConfigView(
|
||||
// store: Store(initialState: ConfigFeature.State()) {
|
||||
// ConfigFeature()
|
||||
// }
|
||||
// )
|
||||
//}
|
||||
|
||||
@@ -16,21 +16,42 @@ struct CreateFeedFeature {
|
||||
processedImages.count < 9
|
||||
}
|
||||
var canPublish: Bool {
|
||||
!content.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty && !isLoading
|
||||
(!content.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty || !processedImages.isEmpty) && !isLoading
|
||||
}
|
||||
var isLoading: Bool = false
|
||||
|
||||
// 新增:图片上传相关状态
|
||||
var uploadedImageUrls: [String] = []
|
||||
var uploadedImages: [UIImage] = [] // 保存原始图片用于获取尺寸信息
|
||||
var isUploadingImages: Bool = false
|
||||
var uploadProgress: Double = 0.0
|
||||
var uploadStatus: String = ""
|
||||
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
}
|
||||
|
||||
enum Action {
|
||||
case contentChanged(String)
|
||||
case publishButtonTapped
|
||||
case publishResponse(Result<PublishDynamicResponse, Error>)
|
||||
case publishResponse(Result<PublishFeedResponse, Error>)
|
||||
case clearError
|
||||
case dismissView
|
||||
case photosPickerItemsChanged([PhotosPickerItem])
|
||||
case processPhotosPickerItems([PhotosPickerItem])
|
||||
case removeImage(Int)
|
||||
case updateProcessedImages([UIImage])
|
||||
|
||||
// 新增:图片上传相关 Action
|
||||
case uploadImagesToCOS
|
||||
case imageUploadProgress(Double, Int, Int) // progress, current, total
|
||||
case imageUploadCompleted([String], [UIImage]) // urls, images
|
||||
case imageUploadFailed(Error)
|
||||
case publishContent
|
||||
|
||||
// 新增:发布成功通知
|
||||
case publishSuccess
|
||||
}
|
||||
|
||||
@Dependency(\.apiService) var apiService
|
||||
@@ -44,11 +65,13 @@ struct CreateFeedFeature {
|
||||
state.content = newContent
|
||||
state.characterCount = newContent.count
|
||||
return .none
|
||||
|
||||
case .photosPickerItemsChanged(let items):
|
||||
state.selectedImages = items
|
||||
return .run { send in
|
||||
await send(.processPhotosPickerItems(items))
|
||||
}
|
||||
|
||||
case .processPhotosPickerItems(let items):
|
||||
let currentImages = state.processedImages
|
||||
return .run { send in
|
||||
@@ -60,63 +83,180 @@ struct CreateFeedFeature {
|
||||
newImages.append(image)
|
||||
}
|
||||
}
|
||||
await MainActor.run {
|
||||
send(.updateProcessedImages(newImages))
|
||||
}
|
||||
await send(.updateProcessedImages(newImages))
|
||||
}
|
||||
|
||||
case .updateProcessedImages(let images):
|
||||
state.processedImages = images
|
||||
// 清空之前的上传结果
|
||||
state.uploadedImageUrls = []
|
||||
return .none
|
||||
|
||||
case .removeImage(let index):
|
||||
guard index < state.processedImages.count else { return .none }
|
||||
state.processedImages.remove(at: index)
|
||||
if index < state.selectedImages.count {
|
||||
state.selectedImages.remove(at: index)
|
||||
}
|
||||
// 同时移除对应的上传链接
|
||||
if index < state.uploadedImageUrls.count {
|
||||
state.uploadedImageUrls.remove(at: index)
|
||||
}
|
||||
return .none
|
||||
|
||||
case .publishButtonTapped:
|
||||
guard state.canPublish else {
|
||||
state.errorMessage = "请输入内容"
|
||||
state.errorMessage = "请输入内容或选择图片"
|
||||
return .none
|
||||
}
|
||||
|
||||
// 如果有图片且还没有上传,先上传图片
|
||||
if !state.processedImages.isEmpty && state.uploadedImageUrls.isEmpty {
|
||||
return .send(.uploadImagesToCOS)
|
||||
}
|
||||
|
||||
// 直接发布(图片已上传或没有图片)
|
||||
return .send(.publishContent)
|
||||
|
||||
case .uploadImagesToCOS:
|
||||
guard !state.processedImages.isEmpty else {
|
||||
return .send(.publishContent)
|
||||
}
|
||||
|
||||
state.isUploadingImages = true
|
||||
state.uploadProgress = 0.0
|
||||
state.uploadStatus = "正在上传图片..."
|
||||
state.errorMessage = nil
|
||||
|
||||
// 提取状态值到局部变量,避免在 @Sendable 闭包中访问 inout 参数
|
||||
let imagesToUpload = state.processedImages
|
||||
|
||||
return .run { send in
|
||||
var uploadedUrls: [String] = []
|
||||
var uploadedImages: [UIImage] = []
|
||||
let totalImages = imagesToUpload.count
|
||||
|
||||
for (index, image) in imagesToUpload.enumerated() {
|
||||
// 更新上传进度
|
||||
await send(.imageUploadProgress(Double(index) / Double(totalImages), index + 1, totalImages))
|
||||
|
||||
// 上传图片到 COS
|
||||
if let imageUrl = await COSManager.shared.uploadUIImage(image, apiService: apiService) {
|
||||
uploadedUrls.append(imageUrl)
|
||||
uploadedImages.append(image) // 保存原始图片
|
||||
} else {
|
||||
// 上传失败
|
||||
await send(.imageUploadFailed(APIError.custom("图片上传失败")))
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
// 所有图片上传完成
|
||||
await send(.imageUploadProgress(1.0, totalImages, totalImages))
|
||||
await send(.imageUploadCompleted(uploadedUrls, uploadedImages))
|
||||
}
|
||||
|
||||
case .imageUploadProgress(let progress, let current, let total):
|
||||
state.uploadProgress = progress
|
||||
state.uploadStatus = "正在上传图片... (\(current)/\(total))"
|
||||
return .none
|
||||
|
||||
case .imageUploadCompleted(let urls, let images):
|
||||
state.isUploadingImages = false
|
||||
state.uploadedImageUrls = urls
|
||||
state.uploadedImages = images
|
||||
state.uploadStatus = "图片上传完成"
|
||||
// 上传完成后自动发布内容
|
||||
return .send(.publishContent)
|
||||
|
||||
case .imageUploadFailed(let error):
|
||||
state.isUploadingImages = false
|
||||
state.errorMessage = "图片上传失败: \(error.localizedDescription)"
|
||||
return .none
|
||||
|
||||
case .publishContent:
|
||||
state.isLoading = true
|
||||
state.errorMessage = nil
|
||||
let request = PublishDynamicRequest(
|
||||
content: state.content.trimmingCharacters(in: .whitespacesAndNewlines),
|
||||
images: state.processedImages
|
||||
)
|
||||
|
||||
// 提取状态值到局部变量,避免在 @Sendable 闭包中访问 inout 参数
|
||||
let content = state.content.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
let imageUrls = state.uploadedImageUrls
|
||||
let images = state.uploadedImages
|
||||
|
||||
return .run { send in
|
||||
do {
|
||||
// 构建 ResListItem 数组
|
||||
var resList: [ResListItem] = []
|
||||
for (index, imageUrl) in imageUrls.enumerated() {
|
||||
if index < images.count, let cgImage = images[index].cgImage {
|
||||
let width = cgImage.width
|
||||
let height = cgImage.height
|
||||
let format = "jpeg"
|
||||
let item = ResListItem(resUrl: imageUrl, width: width, height: height, format: format)
|
||||
resList.append(item)
|
||||
}
|
||||
}
|
||||
|
||||
// 使用 PublishFeedRequest 而不是 PublishDynamicRequest
|
||||
let userId = await UserInfoManager.getCurrentUserId() ?? ""
|
||||
let type = resList.isEmpty ? "0" : "2" // 0: 纯文字, 2: 图片
|
||||
let request = await PublishFeedRequest.make(
|
||||
content: content.isEmpty ? "" : content,
|
||||
uid: userId,
|
||||
type: type,
|
||||
resList: resList.isEmpty ? nil : resList
|
||||
)
|
||||
|
||||
let response = try await apiService.request(request)
|
||||
await send(.publishResponse(.success(response)))
|
||||
} catch {
|
||||
await send(.publishResponse(.failure(error)))
|
||||
}
|
||||
}
|
||||
|
||||
case .publishResponse(.success(let response)):
|
||||
state.isLoading = false
|
||||
if response.code == 200 {
|
||||
return .send(.dismissView)
|
||||
// 发布成功,先发送通知,然后关闭页面
|
||||
return .merge(
|
||||
.send(.publishSuccess),
|
||||
.send(.dismissView)
|
||||
)
|
||||
} else {
|
||||
state.errorMessage = response.message.isEmpty ? "发布失败" : response.message
|
||||
return .none
|
||||
}
|
||||
|
||||
case .publishResponse(.failure(let error)):
|
||||
state.isLoading = false
|
||||
state.errorMessage = error.localizedDescription
|
||||
return .none
|
||||
|
||||
case .clearError:
|
||||
state.errorMessage = nil
|
||||
return .none
|
||||
|
||||
case .dismissView:
|
||||
// 检查是否在presentation context中
|
||||
guard isPresented else {
|
||||
// 如果不在presentation context中,不执行dismiss
|
||||
return .none
|
||||
}
|
||||
// 始终发送通知,让外层处理关闭逻辑
|
||||
return .run { _ in
|
||||
await dismiss()
|
||||
await MainActor.run {
|
||||
NotificationCenter.default.post(name: .init("CreateFeedDismiss"), object: nil)
|
||||
}
|
||||
}
|
||||
case .publishSuccess:
|
||||
// 发送通知给外层刷新列表和关闭页面
|
||||
return .merge(
|
||||
.run { _ in
|
||||
await MainActor.run {
|
||||
NotificationCenter.default.post(name: .init("CreateFeedPublishSuccess"), object: nil)
|
||||
}
|
||||
},
|
||||
.run { _ in
|
||||
await MainActor.run {
|
||||
NotificationCenter.default.post(name: .init("CreateFeedDismiss"), object: nil)
|
||||
}
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -135,6 +275,18 @@ extension CreateFeedFeature.Action: Equatable {
|
||||
return true
|
||||
case let (.removeImage(a), .removeImage(b)):
|
||||
return a == b
|
||||
case (.uploadImagesToCOS, .uploadImagesToCOS):
|
||||
return true
|
||||
case let (.imageUploadProgress(a, b, c), .imageUploadProgress(d, e, f)):
|
||||
return a == d && b == e && c == f
|
||||
case let (.imageUploadCompleted(a, c), .imageUploadCompleted(b, d)):
|
||||
return a == b && c.count == d.count // 简化比较,只比较URL数组和图片数量
|
||||
case let (.imageUploadFailed(a), .imageUploadFailed(b)):
|
||||
return a.localizedDescription == b.localizedDescription
|
||||
case (.publishContent, .publishContent):
|
||||
return true
|
||||
case (.publishSuccess, .publishSuccess):
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
@@ -143,43 +295,5 @@ extension CreateFeedFeature.Action: Equatable {
|
||||
|
||||
// MARK: - 发布动态相关模型
|
||||
|
||||
struct PublishDynamicRequest: APIRequestProtocol {
|
||||
typealias Response = PublishDynamicResponse
|
||||
let endpoint: String = APIEndpoint.publishFeed.path
|
||||
let method: HTTPMethod = .POST
|
||||
let includeBaseParameters: Bool = true
|
||||
let queryParameters: [String: String]? = nil
|
||||
let timeout: TimeInterval = 30.0
|
||||
let content: String
|
||||
let images: [UIImage]
|
||||
let type: Int // 0: 纯文字, 2: 图片
|
||||
init(content: String, images: [UIImage] = []) {
|
||||
self.content = content
|
||||
self.images = images
|
||||
self.type = images.isEmpty ? 0 : 2
|
||||
}
|
||||
var bodyParameters: [String: Any]? {
|
||||
var params: [String: Any] = [
|
||||
"content": content,
|
||||
"type": type
|
||||
]
|
||||
if !images.isEmpty {
|
||||
let imageData = images.compactMap { image in
|
||||
image.jpegData(compressionQuality: 0.8)?.base64EncodedString()
|
||||
}
|
||||
params["images"] = imageData
|
||||
}
|
||||
return params
|
||||
}
|
||||
}
|
||||
|
||||
struct PublishDynamicResponse: Codable {
|
||||
let code: Int
|
||||
let message: String
|
||||
let data: PublishDynamicData?
|
||||
}
|
||||
|
||||
struct PublishDynamicData: Codable {
|
||||
let dynamicId: Int
|
||||
let publishTime: Int
|
||||
}
|
||||
// 注意:现在使用 DynamicsModels.swift 中的 PublishFeedRequest 和 PublishFeedResponse
|
||||
// 不再需要重复定义这些模型
|
||||
|
||||
213
yana/Features/DetailFeature.swift
Normal file
213
yana/Features/DetailFeature.swift
Normal file
@@ -0,0 +1,213 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
@Reducer
|
||||
struct DetailFeature {
|
||||
@Dependency(\.apiService) var apiService
|
||||
@Dependency(\.isPresented) var isPresented
|
||||
|
||||
@ObservableState
|
||||
struct State: Equatable {
|
||||
var moment: MomentsInfo
|
||||
var isLikeLoading = false
|
||||
var isDeleteLoading = false
|
||||
var showImagePreview = false
|
||||
var selectedImageIndex = 0
|
||||
var selectedImages: [String] = []
|
||||
|
||||
// 新增:当前用户ID状态
|
||||
var currentUserId: String?
|
||||
var isLoadingCurrentUserId = false
|
||||
|
||||
// 新增:是否需要关闭DetailView
|
||||
var shouldDismiss = false
|
||||
|
||||
// 新增:显示用户主页相关状态
|
||||
var showUserProfile = false
|
||||
var targetUserId: Int = 0
|
||||
|
||||
init(moment: MomentsInfo) {
|
||||
self.moment = moment
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
case onAppear
|
||||
case likeDynamic(Int, Int, Int, Int) // dynamicId, uid, likedUid, worldId
|
||||
case likeResponse(TaskResult<LikeDynamicResponse>)
|
||||
case deleteDynamic
|
||||
case deleteResponse(TaskResult<DeleteDynamicResponse>)
|
||||
case showImagePreview([String], Int)
|
||||
case hideImagePreview
|
||||
case imagePreviewDismissed
|
||||
case dismissView
|
||||
|
||||
// 新增:当前用户ID相关actions
|
||||
case loadCurrentUserId
|
||||
case currentUserIdLoaded(String?)
|
||||
|
||||
// 新增:用户主页相关actions
|
||||
case showUserProfile(Int)
|
||||
case hideUserProfile
|
||||
}
|
||||
|
||||
var body: some ReducerOf<Self> {
|
||||
Reduce {
|
||||
state,
|
||||
action in
|
||||
switch action {
|
||||
case .onAppear:
|
||||
// 如果还没有获取过当前用户ID,则开始获取
|
||||
if state.currentUserId == nil && !state.isLoadingCurrentUserId {
|
||||
return .send(.loadCurrentUserId)
|
||||
}
|
||||
return .none
|
||||
|
||||
case .loadCurrentUserId:
|
||||
state.isLoadingCurrentUserId = true
|
||||
return .run { send in
|
||||
let userId = await UserInfoManager.getCurrentUserId()
|
||||
debugInfoSync("🔍 DetailFeature: 获取当前用户ID - \(userId ?? "nil")")
|
||||
await send(.currentUserIdLoaded(userId))
|
||||
}
|
||||
|
||||
case let .currentUserIdLoaded(userId):
|
||||
state.currentUserId = userId
|
||||
state.isLoadingCurrentUserId = false
|
||||
debugInfoSync("✅ DetailFeature: 当前用户ID已加载 - \(userId ?? "nil")")
|
||||
return .none
|
||||
|
||||
case let .likeDynamic(dynamicId, uid, likedUid, worldId):
|
||||
// 设置loading状态
|
||||
state.isLikeLoading = true
|
||||
|
||||
let status = state.moment.isLike ? 0 : 1 // 0: 取消点赞, 1: 点赞
|
||||
let request = LikeDynamicRequest(
|
||||
dynamicId: dynamicId,
|
||||
uid: uid,
|
||||
status: status,
|
||||
likedUid: likedUid,
|
||||
worldId: worldId
|
||||
)
|
||||
|
||||
return .run { [apiService] send in
|
||||
do {
|
||||
let response: LikeDynamicResponse = try await apiService.request(request)
|
||||
await send(.likeResponse(.success(response)))
|
||||
} catch {
|
||||
await send(.likeResponse(.failure(error)))
|
||||
}
|
||||
}
|
||||
|
||||
case let .likeResponse(.success(response)):
|
||||
if let data = response.data, let success = data.success, success {
|
||||
// 根据API响应更新点赞状态
|
||||
let newLikeState = !state.moment.isLike // 切换点赞状态
|
||||
|
||||
// 创建更新后的动态对象
|
||||
let updatedMoment = MomentsInfo(
|
||||
dynamicId: state.moment.dynamicId,
|
||||
uid: state.moment.uid,
|
||||
nick: state.moment.nick,
|
||||
avatar: state.moment.avatar,
|
||||
type: state.moment.type,
|
||||
content: state.moment.content,
|
||||
likeCount: data.likeCount ?? state.moment.likeCount,
|
||||
isLike: newLikeState,
|
||||
commentCount: state.moment.commentCount,
|
||||
publishTime: state.moment.publishTime,
|
||||
worldId: state.moment.worldId,
|
||||
status: state.moment.status,
|
||||
playCount: state.moment.playCount,
|
||||
dynamicResList: state.moment.dynamicResList,
|
||||
gender: state.moment.gender,
|
||||
squareTop: state.moment.squareTop,
|
||||
topicTop: state.moment.topicTop,
|
||||
newUser: state.moment.newUser,
|
||||
defUser: state.moment.defUser,
|
||||
scene: state.moment.scene,
|
||||
userVipInfoVO: state.moment.userVipInfoVO,
|
||||
headwearPic: state.moment.headwearPic,
|
||||
headwearEffect: state.moment.headwearEffect,
|
||||
headwearType: state.moment.headwearType,
|
||||
headwearName: state.moment.headwearName,
|
||||
headwearId: state.moment.headwearId,
|
||||
experLevelPic: state.moment.experLevelPic,
|
||||
charmLevelPic: state.moment.charmLevelPic,
|
||||
isCustomWord: state.moment.isCustomWord,
|
||||
labelList: state.moment.labelList
|
||||
)
|
||||
state.moment = updatedMoment
|
||||
// 移除loading状态
|
||||
state.isLikeLoading = false
|
||||
} else {
|
||||
// API返回失败,通过APILoadingManager显示错误信息
|
||||
let errorMessage = response.message.isEmpty ? "点赞失败,请重试" : response.message
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: errorMessage)
|
||||
}
|
||||
|
||||
// 移除loading状态
|
||||
state.isLikeLoading = false
|
||||
return .none
|
||||
|
||||
case let .likeResponse(.failure(error)):
|
||||
// 移除loading状态
|
||||
state.isLikeLoading = false
|
||||
// 通过APILoadingManager显示错误信息
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: error.localizedDescription)
|
||||
return .none
|
||||
|
||||
case .deleteDynamic:
|
||||
state.isDeleteLoading = true
|
||||
|
||||
let request = DeleteDynamicRequest(dynamicId: state.moment.dynamicId, uid: state.moment.uid)
|
||||
|
||||
return .run { send in
|
||||
let result = await TaskResult {
|
||||
try await apiService.request(request)
|
||||
}
|
||||
await send(.deleteResponse(result))
|
||||
}
|
||||
|
||||
case let .deleteResponse(.success(response)):
|
||||
state.isDeleteLoading = false
|
||||
debugInfoSync("✅ DetailFeature: 动态删除成功")
|
||||
// 删除成功,返回上一页
|
||||
return .send(.dismissView)
|
||||
|
||||
case let .deleteResponse(.failure(error)):
|
||||
state.isDeleteLoading = false
|
||||
// 可以在这里处理错误
|
||||
return .none
|
||||
|
||||
case let .showImagePreview(images, index):
|
||||
state.selectedImages = images
|
||||
state.selectedImageIndex = index
|
||||
state.showImagePreview = true
|
||||
return .none
|
||||
|
||||
case .hideImagePreview:
|
||||
state.showImagePreview = false
|
||||
return .none
|
||||
|
||||
case .imagePreviewDismissed:
|
||||
state.showImagePreview = false
|
||||
return .none
|
||||
|
||||
case .dismissView:
|
||||
debugInfoSync("🔍 DetailFeature: 请求关闭DetailView")
|
||||
state.shouldDismiss = true
|
||||
return .none
|
||||
|
||||
case let .showUserProfile(userId):
|
||||
state.targetUserId = userId
|
||||
state.showUserProfile = true
|
||||
return .none
|
||||
|
||||
case .hideUserProfile:
|
||||
state.showUserProfile = false
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -20,13 +20,11 @@ struct EMailLoginFeature {
|
||||
case failed
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
init() {
|
||||
self.email = "exzero@126.com"
|
||||
self.email = ""
|
||||
self.verificationCode = ""
|
||||
self.loginStep = .initial
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
enum Action {
|
||||
@@ -57,12 +55,12 @@ struct EMailLoginFeature {
|
||||
|
||||
case .getVerificationCodeTapped:
|
||||
guard !state.email.isEmpty else {
|
||||
state.errorMessage = NSLocalizedString("email_login.email_required", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("email_login.email_required", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
guard ValidationHelper.isValidEmail(state.email) else {
|
||||
state.errorMessage = NSLocalizedString("email_login.invalid_email", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("email_login.invalid_email", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
@@ -107,12 +105,12 @@ struct EMailLoginFeature {
|
||||
|
||||
case .loginButtonTapped(let email, let verificationCode):
|
||||
guard !email.isEmpty && !verificationCode.isEmpty else {
|
||||
state.errorMessage = NSLocalizedString("email_login.fields_required", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("email_login.fields_required", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
guard ValidationHelper.isValidEmail(email) else {
|
||||
state.errorMessage = NSLocalizedString("email_login.invalid_email", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("email_login.invalid_email", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
|
||||
@@ -25,6 +25,20 @@ struct EditFeedFeature {
|
||||
var isUploadingImages: Bool = false
|
||||
var imageUploadProgress: Double = 0.0 // 0.0~1.0
|
||||
var uploadedResList: [ResListItem] = []
|
||||
|
||||
// 新增:PhotosPicker相关状态
|
||||
var showPhotosPicker: Bool = false
|
||||
var selectedPhotoItems: [PhotosPickerItem] = []
|
||||
|
||||
// 新增:删除图片确认相关状态
|
||||
var showDeleteImageAlert: Bool = false
|
||||
var imageToDeleteIndex: Int? = nil
|
||||
|
||||
// 默认初始化器
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
|
||||
// 手动实现Equatable,selectedImages只比较数量(PhotosPickerItem不支持Equatable)
|
||||
static func == (lhs: State, rhs: State) -> Bool {
|
||||
lhs.content == rhs.content &&
|
||||
@@ -35,7 +49,11 @@ struct EditFeedFeature {
|
||||
lhs.selectedImages.count == rhs.selectedImages.count &&
|
||||
lhs.isUploadingImages == rhs.isUploadingImages &&
|
||||
lhs.imageUploadProgress == rhs.imageUploadProgress &&
|
||||
lhs.uploadedResList == rhs.uploadedResList
|
||||
lhs.uploadedResList == rhs.uploadedResList &&
|
||||
lhs.showPhotosPicker == rhs.showPhotosPicker &&
|
||||
lhs.selectedPhotoItems.count == rhs.selectedPhotoItems.count &&
|
||||
lhs.showDeleteImageAlert == rhs.showDeleteImageAlert &&
|
||||
lhs.imageToDeleteIndex == rhs.imageToDeleteIndex
|
||||
}
|
||||
}
|
||||
|
||||
@@ -56,6 +74,12 @@ struct EditFeedFeature {
|
||||
case uploadImagesResponse(Result<[ResListItem], Error>)
|
||||
// 新增:图片上传进度
|
||||
case updateImageUploadProgress(Double)
|
||||
// 新增:PhotosPicker相关Action
|
||||
case photosPickerDismissed
|
||||
case addImageButtonTapped
|
||||
// 新增:删除图片确认相关Action
|
||||
case showDeleteImageAlert(Int)
|
||||
case deleteImageAlertDismissed
|
||||
}
|
||||
|
||||
@Dependency(\.apiService) var apiService
|
||||
@@ -176,6 +200,7 @@ struct EditFeedFeature {
|
||||
return .none
|
||||
case .photosPickerItemsChanged(let items):
|
||||
state.selectedImages = items
|
||||
state.selectedPhotoItems = items
|
||||
return .run { send in
|
||||
await send(.processPhotosPickerItems(items))
|
||||
}
|
||||
@@ -203,12 +228,31 @@ struct EditFeedFeature {
|
||||
if index < state.selectedImages.count {
|
||||
state.selectedImages.remove(at: index)
|
||||
}
|
||||
if index < state.selectedPhotoItems.count {
|
||||
state.selectedPhotoItems.remove(at: index)
|
||||
}
|
||||
return .none
|
||||
// 新增:图片上传进度
|
||||
case .updateImageUploadProgress(let progress):
|
||||
state.imageUploadProgress = progress
|
||||
return .none
|
||||
// 新增:PhotosPicker相关Action
|
||||
case .photosPickerDismissed:
|
||||
state.showPhotosPicker = false
|
||||
return .none
|
||||
case .addImageButtonTapped:
|
||||
state.showPhotosPicker = true
|
||||
return .none
|
||||
// 新增:删除图片确认相关Action
|
||||
case .showDeleteImageAlert(let index):
|
||||
state.imageToDeleteIndex = index
|
||||
state.showDeleteImageAlert = true
|
||||
return .none
|
||||
case .deleteImageAlertDismissed:
|
||||
state.showDeleteImageAlert = false
|
||||
state.imageToDeleteIndex = nil
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,111 +0,0 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
@Reducer
|
||||
struct FeedFeature {
|
||||
@ObservableState
|
||||
struct State: Equatable {
|
||||
var moments: [MomentsInfo] = []
|
||||
var isLoading = false
|
||||
var isRefreshing = false
|
||||
var hasMoreData = true
|
||||
var error: String?
|
||||
var nextDynamicId: Int = 0
|
||||
// CreateFeedView 相关状态
|
||||
var createFeedState = CreateFeedFeature.State()
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
case onAppear
|
||||
case refresh
|
||||
case loadLatestMoments
|
||||
case loadMoreMoments
|
||||
case momentsResponse(TaskResult<MomentsLatestResponse>)
|
||||
case clearError
|
||||
case retryLoad
|
||||
// CreateFeedView 相关 Action
|
||||
case createFeedCompleted
|
||||
case createFeedDismissed
|
||||
// CreateFeedFeature 的 action
|
||||
case createFeed(CreateFeedFeature.Action)
|
||||
}
|
||||
|
||||
@Dependency(\.apiService) var apiService
|
||||
|
||||
var body: some ReducerOf<Self> {
|
||||
Scope(state: \.createFeedState, action: \.createFeed) {
|
||||
CreateFeedFeature()
|
||||
}
|
||||
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .onAppear:
|
||||
guard state.moments.isEmpty && !state.isLoading else { return .none }
|
||||
return .send(.loadLatestMoments)
|
||||
case .refresh:
|
||||
guard !state.isRefreshing else { return .none }
|
||||
state.isRefreshing = true
|
||||
state.error = nil
|
||||
let request = LatestDynamicsRequest(dynamicId: "", pageSize: 20, types: [.text, .picture])
|
||||
return .run { send in
|
||||
await send(.momentsResponse(TaskResult { try await apiService.request(request) }))
|
||||
}
|
||||
case .loadLatestMoments:
|
||||
guard !state.isLoading else { return .none }
|
||||
state.isLoading = true
|
||||
state.error = nil
|
||||
let request = LatestDynamicsRequest(dynamicId: "", pageSize: 20, types: [.text, .picture])
|
||||
return .run { send in
|
||||
await send(.momentsResponse(TaskResult { try await apiService.request(request) }))
|
||||
}
|
||||
case .loadMoreMoments:
|
||||
guard !state.isLoading && state.hasMoreData else { return .none }
|
||||
state.isLoading = true
|
||||
state.error = nil
|
||||
let request = LatestDynamicsRequest(dynamicId: state.nextDynamicId == 0 ? "" : String(state.nextDynamicId), pageSize: 20, types: [.text, .picture])
|
||||
return .run { send in
|
||||
await send(.momentsResponse(TaskResult { try await apiService.request(request) }))
|
||||
}
|
||||
case let .momentsResponse(.success(response)):
|
||||
state.isLoading = false
|
||||
state.isRefreshing = false
|
||||
guard response.code == 200, let data = response.data else {
|
||||
let errorMsg = response.message.isEmpty ? "获取动态失败" : response.message
|
||||
state.error = errorMsg
|
||||
return .none
|
||||
}
|
||||
let isRefresh = state.nextDynamicId == 0 || state.isRefreshing
|
||||
if isRefresh {
|
||||
state.moments = data.dynamicList
|
||||
} else {
|
||||
state.moments.append(contentsOf: data.dynamicList)
|
||||
}
|
||||
state.nextDynamicId = data.nextDynamicId
|
||||
state.hasMoreData = !data.dynamicList.isEmpty
|
||||
return .none
|
||||
case let .momentsResponse(.failure(error)):
|
||||
state.isLoading = false
|
||||
state.isRefreshing = false
|
||||
state.error = error.localizedDescription
|
||||
return .none
|
||||
case .clearError:
|
||||
state.error = nil
|
||||
return .none
|
||||
case .retryLoad:
|
||||
if state.moments.isEmpty {
|
||||
return .send(.loadLatestMoments)
|
||||
} else {
|
||||
return .send(.loadMoreMoments)
|
||||
}
|
||||
case .createFeedCompleted:
|
||||
return .send(.refresh)
|
||||
case .createFeedDismissed:
|
||||
return .none
|
||||
case .createFeed(.dismissView):
|
||||
return .send(.createFeedDismissed)
|
||||
case .createFeed:
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -4,12 +4,13 @@ import ComposableArchitecture
|
||||
@Reducer
|
||||
struct FeedListFeature {
|
||||
@Dependency(\.apiService) var apiService
|
||||
@ObservableState
|
||||
struct State: Equatable {
|
||||
var isFirstLoad: Bool = true
|
||||
var feeds: [Feed] = [] // 预留 feed 内容
|
||||
var isLoading: Bool = false
|
||||
var error: String? = nil
|
||||
var isEditFeedPresented: Bool = false // 新增:控制 EditFeedView 弹窗
|
||||
var isEditFeedPresented: Bool = false // 新增:控制 CreateFeedView 弹窗
|
||||
// 新增:动态内容
|
||||
var moments: [MomentsInfo] = []
|
||||
// 新增:只加载一次标志
|
||||
@@ -18,6 +19,15 @@ struct FeedListFeature {
|
||||
var currentPage: Int = 1
|
||||
var hasMore: Bool = true
|
||||
var isLoadingMore: Bool = false
|
||||
// 新增:DetailView相关状态
|
||||
var showDetail: Bool = false
|
||||
var selectedMoment: MomentsInfo?
|
||||
// 新增:点赞相关状态
|
||||
var likeLoadingDynamicIds: Set<Int> = []
|
||||
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
@@ -31,7 +41,16 @@ struct FeedListFeature {
|
||||
// 新增:动态内容相关
|
||||
case fetchFeeds
|
||||
case fetchFeedsResponse(TaskResult<MomentsLatestResponse>)
|
||||
// 新增:DetailView相关Action
|
||||
case showDetail(MomentsInfo)
|
||||
case detailDismissed
|
||||
// 新增:点赞相关Action
|
||||
case likeDynamic(Int, Int, Int, Int) // dynamicId, uid, likedUid, worldId
|
||||
case likeResponse(TaskResult<LikeDynamicResponse>, dynamicId: Int, loadingId: UUID?)
|
||||
// 新增:CreateFeed发布成功通知
|
||||
case createFeedPublishSuccess
|
||||
// 预留后续 Action
|
||||
case checkAuthAndLoad
|
||||
}
|
||||
|
||||
func reduce(into state: inout State, action: Action) -> Effect<Action> {
|
||||
@@ -39,7 +58,36 @@ struct FeedListFeature {
|
||||
case .onAppear:
|
||||
guard state.isFirstLoad else { return .none }
|
||||
state.isFirstLoad = false
|
||||
return .send(.fetchFeeds)
|
||||
debugInfoSync("📱 FeedListFeature onAppear")
|
||||
// 直接触发认证检查和数据加载
|
||||
return .send(.checkAuthAndLoad)
|
||||
|
||||
case .checkAuthAndLoad:
|
||||
// 新增:认证检查和数据加载
|
||||
return .run { send in
|
||||
// 检查认证信息是否已保存
|
||||
let accountModel = await UserInfoManager.getAccountModel()
|
||||
if accountModel?.uid != nil {
|
||||
debugInfoSync("✅ FeedListFeature: 认证信息已准备好,开始获取动态")
|
||||
await send(.fetchFeeds)
|
||||
return
|
||||
} else {
|
||||
debugInfoSync("⏳ FeedListFeature: 认证信息未准备好,等待...")
|
||||
// 增加等待时间和重试次数
|
||||
for attempt in 1...3 {
|
||||
try? await Task.sleep(nanoseconds: 500_000_000) // 0.5秒
|
||||
let retryAccountModel = await UserInfoManager.getAccountModel()
|
||||
if retryAccountModel?.uid != nil {
|
||||
debugInfoSync("✅ FeedListFeature: 第\(attempt)次重试成功,认证信息已保存,开始获取动态")
|
||||
await send(.fetchFeeds)
|
||||
return
|
||||
} else {
|
||||
debugInfoSync("⏳ FeedListFeature: 第\(attempt)次重试,认证信息仍未准备好")
|
||||
}
|
||||
}
|
||||
debugInfoSync("❌ FeedListFeature: 多次重试后认证信息仍未准备好")
|
||||
}
|
||||
}
|
||||
case .reload:
|
||||
// 下拉刷新,重置状态并请求第一页
|
||||
state.isLoading = true
|
||||
@@ -94,24 +142,36 @@ struct FeedListFeature {
|
||||
case .fetchFeeds:
|
||||
state.isLoading = true
|
||||
state.error = nil
|
||||
debugInfoSync("🔄 FeedListFeature: 开始获取动态")
|
||||
// 发起 API 请求
|
||||
return .run { [apiService] send in
|
||||
await send(.fetchFeedsResponse(TaskResult {
|
||||
let request = LatestDynamicsRequest(dynamicId: "", pageSize: 20, types: [.text, .picture])
|
||||
debugInfoSync("📡 FeedListFeature: 发送请求: \(request.endpoint)")
|
||||
debugInfoSync(" 参数: dynamicId=\(request.dynamicId), pageSize=\(request.pageSize)")
|
||||
return try await apiService.request(request)
|
||||
}))
|
||||
}
|
||||
case let .fetchFeedsResponse(.success(response)):
|
||||
state.isLoading = false
|
||||
debugInfoSync("✅ FeedListFeature: API 请求成功")
|
||||
debugInfoSync(" 响应码: \(response.code)")
|
||||
debugInfoSync(" 消息: \(response.message)")
|
||||
debugInfoSync(" 数据数量: \(response.data?.dynamicList.count ?? 0)")
|
||||
if let list = response.data?.dynamicList {
|
||||
state.moments = list
|
||||
state.error = nil
|
||||
state.currentPage = 1
|
||||
state.hasMore = (list.count >= 20)
|
||||
debugInfoSync("✅ FeedListFeature: 数据加载成功")
|
||||
debugInfoSync(" 动态数量: \(list.count)")
|
||||
debugInfoSync(" 是否有更多: \(state.hasMore)")
|
||||
} else {
|
||||
state.moments = []
|
||||
state.error = response.message
|
||||
state.hasMore = false
|
||||
debugErrorSync("❌ FeedListFeature: 数据为空")
|
||||
debugErrorSync(" 错误消息: \(response.message)")
|
||||
}
|
||||
return .none
|
||||
case let .fetchFeedsResponse(.failure(error)):
|
||||
@@ -119,6 +179,8 @@ struct FeedListFeature {
|
||||
state.moments = []
|
||||
state.error = error.localizedDescription
|
||||
state.hasMore = false
|
||||
debugErrorSync("❌ FeedListFeature: API 请求失败")
|
||||
debugErrorSync(" 错误: \(error.localizedDescription)")
|
||||
return .none
|
||||
case .editFeedButtonTapped:
|
||||
state.isEditFeedPresented = true
|
||||
@@ -126,9 +188,116 @@ struct FeedListFeature {
|
||||
case .editFeedDismissed:
|
||||
state.isEditFeedPresented = false
|
||||
return .none
|
||||
case .createFeedPublishSuccess:
|
||||
// CreateFeed发布成功,触发刷新并关闭编辑页面
|
||||
return .merge(
|
||||
.send(.reload),
|
||||
.send(.editFeedDismissed)
|
||||
)
|
||||
case .testButtonTapped:
|
||||
debugInfoSync("[LOG] FeedListFeature testButtonTapped")
|
||||
return .none
|
||||
case let .showDetail(moment):
|
||||
state.selectedMoment = moment
|
||||
state.showDetail = true
|
||||
return .none
|
||||
case .detailDismissed:
|
||||
state.showDetail = false
|
||||
state.selectedMoment = nil
|
||||
return .none
|
||||
case let .likeDynamic(dynamicId, uid, likedUid, worldId):
|
||||
// 添加loading状态
|
||||
state.likeLoadingDynamicIds.insert(dynamicId)
|
||||
|
||||
// 找到对应的动态并获取当前点赞状态
|
||||
guard let index = state.moments.firstIndex(where: { $0.dynamicId == dynamicId }) else {
|
||||
// 找不到对应的动态,显示错误信息
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: "找不到对应的动态")
|
||||
state.likeLoadingDynamicIds.remove(dynamicId)
|
||||
return .none
|
||||
}
|
||||
|
||||
let currentMoment = state.moments[index]
|
||||
let status = currentMoment.isLike ? 0 : 1 // 0: 取消点赞, 1: 点赞
|
||||
|
||||
let request = LikeDynamicRequest(
|
||||
dynamicId: dynamicId,
|
||||
uid: uid,
|
||||
status: status,
|
||||
likedUid: likedUid,
|
||||
worldId: worldId
|
||||
)
|
||||
|
||||
return .run { [apiService] send in
|
||||
let loadingId = await APILoadingManager.shared.startLoading(
|
||||
shouldShowLoading: request.shouldShowLoading,
|
||||
shouldShowError: request.shouldShowError
|
||||
)
|
||||
do {
|
||||
let response: LikeDynamicResponse = try await apiService.request(request)
|
||||
await send(.likeResponse(.success(response), dynamicId: dynamicId, loadingId: loadingId))
|
||||
} catch {
|
||||
await send(.likeResponse(.failure(error), dynamicId: dynamicId, loadingId: loadingId))
|
||||
}
|
||||
}
|
||||
|
||||
case let .likeResponse(.success(response), dynamicId, loadingId):
|
||||
state.likeLoadingDynamicIds.remove(dynamicId)
|
||||
if let loadingId = loadingId {
|
||||
if let data = response.data, let success = data.success, success {
|
||||
Task { @MainActor in
|
||||
APILoadingManager.shared.finishLoading(loadingId)
|
||||
}
|
||||
if let index = state.moments.firstIndex(where: { $0.dynamicId == dynamicId }) {
|
||||
let currentMoment = state.moments[index]
|
||||
let newLikeState = !currentMoment.isLike
|
||||
let updatedMoment = MomentsInfo(
|
||||
dynamicId: currentMoment.dynamicId,
|
||||
uid: currentMoment.uid,
|
||||
nick: currentMoment.nick,
|
||||
avatar: currentMoment.avatar,
|
||||
type: currentMoment.type,
|
||||
content: currentMoment.content,
|
||||
likeCount: data.likeCount ?? currentMoment.likeCount,
|
||||
isLike: newLikeState,
|
||||
commentCount: currentMoment.commentCount,
|
||||
publishTime: currentMoment.publishTime,
|
||||
worldId: currentMoment.worldId,
|
||||
status: currentMoment.status,
|
||||
playCount: currentMoment.playCount,
|
||||
dynamicResList: currentMoment.dynamicResList,
|
||||
gender: currentMoment.gender,
|
||||
squareTop: currentMoment.squareTop,
|
||||
topicTop: currentMoment.topicTop,
|
||||
newUser: currentMoment.newUser,
|
||||
defUser: currentMoment.defUser,
|
||||
scene: currentMoment.scene,
|
||||
userVipInfoVO: currentMoment.userVipInfoVO,
|
||||
headwearPic: currentMoment.headwearPic,
|
||||
headwearEffect: currentMoment.headwearEffect,
|
||||
headwearType: currentMoment.headwearType,
|
||||
headwearName: currentMoment.headwearName,
|
||||
headwearId: currentMoment.headwearId,
|
||||
experLevelPic: currentMoment.experLevelPic,
|
||||
charmLevelPic: currentMoment.charmLevelPic,
|
||||
isCustomWord: currentMoment.isCustomWord,
|
||||
labelList: currentMoment.labelList
|
||||
)
|
||||
state.moments[index] = updatedMoment
|
||||
}
|
||||
} else {
|
||||
let errorMessage = response.message.isEmpty ? "点赞失败,请重试" : response.message
|
||||
setAPILoadingErrorSync(loadingId, errorMessage: errorMessage)
|
||||
}
|
||||
}
|
||||
return .none
|
||||
|
||||
case let .likeResponse(.failure(error), dynamicId, loadingId):
|
||||
state.likeLoadingDynamicIds.remove(dynamicId)
|
||||
if let loadingId = loadingId {
|
||||
setAPILoadingErrorSync(loadingId, errorMessage: error.localizedDescription)
|
||||
}
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -141,4 +310,4 @@ enum Feed: Equatable, Identifiable {
|
||||
case .placeholder(let id): return id
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,92 +0,0 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
struct HomeFeature: Reducer {
|
||||
enum Route: Equatable {
|
||||
case createFeed
|
||||
}
|
||||
|
||||
struct State: Equatable {
|
||||
var isInitialized = false
|
||||
var userInfo: UserInfo?
|
||||
var accountModel: AccountModel?
|
||||
var error: String?
|
||||
var feedState = FeedFeature.State()
|
||||
var meDynamic = MeDynamicFeature.State(uid: 0)
|
||||
var isLoggedOut = false
|
||||
var route: Route? = nil
|
||||
}
|
||||
|
||||
@CasePathable
|
||||
enum Action: Equatable {
|
||||
case onAppear
|
||||
case loadUserInfo
|
||||
case userInfoLoaded(UserInfo?)
|
||||
case loadAccountModel
|
||||
case accountModelLoaded(AccountModel?)
|
||||
case logoutTapped
|
||||
case logout
|
||||
case feed(FeedFeature.Action)
|
||||
case meDynamic(MeDynamicFeature.Action)
|
||||
case logoutCompleted
|
||||
case showCreateFeed
|
||||
case createFeedDismissed
|
||||
}
|
||||
|
||||
var body: some ReducerOf<Self> {
|
||||
Scope(state: \.feedState, action: \.feed) {
|
||||
FeedFeature()
|
||||
}
|
||||
Scope(state: \.meDynamic, action: \.meDynamic) {
|
||||
MeDynamicFeature()
|
||||
}
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .onAppear:
|
||||
guard !state.isInitialized else { return .none }
|
||||
state.isInitialized = true
|
||||
return .concatenate(
|
||||
.send(.loadUserInfo),
|
||||
.send(.loadAccountModel)
|
||||
)
|
||||
case .loadUserInfo:
|
||||
return .run { send in
|
||||
let userInfo = await UserInfoManager.getUserInfo()
|
||||
await send(.userInfoLoaded(userInfo))
|
||||
}
|
||||
case let .userInfoLoaded(userInfo):
|
||||
state.userInfo = userInfo
|
||||
state.meDynamic.uid = userInfo?.uid ?? 0
|
||||
return .none
|
||||
case .loadAccountModel:
|
||||
return .run { send in
|
||||
let accountModel = await UserInfoManager.getAccountModel()
|
||||
await send(.accountModelLoaded(accountModel))
|
||||
}
|
||||
case let .accountModelLoaded(accountModel):
|
||||
state.accountModel = accountModel
|
||||
return .none
|
||||
case .logoutTapped:
|
||||
return .send(.logout)
|
||||
case .logout:
|
||||
return .run { send in
|
||||
await UserInfoManager.clearAllAuthenticationData()
|
||||
await send(.logoutCompleted)
|
||||
}
|
||||
case .logoutCompleted:
|
||||
state.isLoggedOut = true
|
||||
return .none
|
||||
case .feed:
|
||||
return .none
|
||||
case .meDynamic:
|
||||
return .none
|
||||
case .showCreateFeed:
|
||||
state.route = .createFeed
|
||||
return .none
|
||||
case .createFeedDismissed:
|
||||
state.route = nil
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -25,15 +25,15 @@ struct IDLoginFeature {
|
||||
case failed // 认证失败
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
init() {
|
||||
self.userID = "2356814"
|
||||
self.password = "a123456"
|
||||
self.userID = ""
|
||||
self.password = ""
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
case userIDChanged(String)
|
||||
case passwordChanged(String)
|
||||
case togglePasswordVisibility
|
||||
case loginButtonTapped(userID: String, password: String)
|
||||
case forgotPasswordTapped
|
||||
@@ -52,6 +52,12 @@ struct IDLoginFeature {
|
||||
var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case let .userIDChanged(userID):
|
||||
state.userID = userID
|
||||
return .none
|
||||
case let .passwordChanged(password):
|
||||
state.password = password
|
||||
return .none
|
||||
case .togglePasswordVisibility:
|
||||
state.isPasswordVisible.toggle()
|
||||
return .none
|
||||
|
||||
@@ -8,6 +8,10 @@ struct InitFeature {
|
||||
var isLoading = false
|
||||
var response: InitResponse?
|
||||
var error: String?
|
||||
|
||||
init() {
|
||||
// 默认初始化
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
|
||||
@@ -11,8 +11,6 @@ struct LoginFeature {
|
||||
var error: String?
|
||||
var idLoginState = IDLoginFeature.State()
|
||||
var emailLoginState = EMailLoginFeature.State() // 新增:邮箱登录状态
|
||||
// 新增:HomeFeature 状态
|
||||
var homeState = HomeFeature.State()
|
||||
|
||||
// 新增:Account Model 和 Ticket 相关状态
|
||||
var accountModel: AccountModel?
|
||||
@@ -36,13 +34,11 @@ struct LoginFeature {
|
||||
case failed // 认证失败
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
init() {
|
||||
// 移除测试用的硬编码凭据
|
||||
// 默认初始化
|
||||
self.account = ""
|
||||
self.password = ""
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
enum Action {
|
||||
@@ -54,7 +50,6 @@ struct LoginFeature {
|
||||
case idLogin(IDLoginFeature.Action)
|
||||
case emailLogin(EMailLoginFeature.Action) // 新增:邮箱登录action
|
||||
// 新增:HomeFeature action
|
||||
case home(HomeFeature.Action)
|
||||
// 新增:Ticket 相关 actions
|
||||
case requestTicket(accessToken: String)
|
||||
case ticketResponse(TaskResult<TicketResponse>)
|
||||
@@ -72,10 +67,6 @@ struct LoginFeature {
|
||||
Scope(state: \.emailLoginState, action: \.emailLogin) {
|
||||
EMailLoginFeature()
|
||||
}
|
||||
// 新增:HomeFeature 作用域
|
||||
Scope(state: \.homeState, action: \.home) {
|
||||
HomeFeature()
|
||||
}
|
||||
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
@@ -241,8 +232,6 @@ struct LoginFeature {
|
||||
case .emailLogin:
|
||||
// EmailLogin动作由子feature处理
|
||||
return .none
|
||||
case .home(_):
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,13 +12,42 @@ struct MainFeature {
|
||||
struct State: Equatable {
|
||||
var selectedTab: Tab = .feed
|
||||
var feedList: FeedListFeature.State = .init()
|
||||
var me: MeFeature.State = .init()
|
||||
var me: MeFeature.State
|
||||
var accountModel: AccountModel? = nil
|
||||
// 新增:导航路径和设置页面 State
|
||||
var navigationPath: [Destination] = []
|
||||
var appSettingState: AppSettingFeature.State? = nil
|
||||
// 新增:登出标志
|
||||
var isLoggedOut: Bool = false
|
||||
|
||||
init(accountModel: AccountModel? = nil) {
|
||||
self.accountModel = accountModel
|
||||
let uid = accountModel?.uid.flatMap { Int($0) } ?? 0
|
||||
debugInfoSync("🏗️ MainFeature 初始化")
|
||||
debugInfoSync(" accountModel.uid: \(accountModel?.uid ?? "nil")")
|
||||
debugInfoSync(" 转换后的uid: \(uid)")
|
||||
|
||||
// 如果没有传入accountModel,尝试从Keychain获取
|
||||
if accountModel == nil {
|
||||
debugInfoSync(" 🔍 尝试从Keychain获取AccountModel")
|
||||
Task {
|
||||
if let savedAccountModel = await UserInfoManager.getAccountModel() {
|
||||
debugInfoSync(" ✅ 从Keychain获取到AccountModel: \(savedAccountModel.uid ?? "nil")")
|
||||
} else {
|
||||
debugInfoSync(" ⚠️ 从Keychain未获取到AccountModel")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
var meState = MeFeature.State(displayUID: uid > 0 ? uid : nil)
|
||||
if uid > 0 {
|
||||
meState.uid = uid // 确保uid与displayUID一致
|
||||
}
|
||||
self.me = meState
|
||||
debugInfoSync(" meState.uid: \(meState.uid)")
|
||||
debugInfoSync(" meState.displayUID: \(meState.displayUID ?? -1)")
|
||||
debugInfoSync(" meState.effectiveUID: \(meState.effectiveUID)")
|
||||
}
|
||||
}
|
||||
|
||||
// 新增:导航目标
|
||||
@@ -57,23 +86,75 @@ struct MainFeature {
|
||||
await send(.accountModelLoaded(accountModel))
|
||||
}
|
||||
case .selectTab(let tab):
|
||||
debugInfoSync("🎯 MainFeature selectTab: \(tab)")
|
||||
debugInfoSync(" 当前selectedTab: \(state.selectedTab)")
|
||||
debugInfoSync(" 新selectedTab: \(tab)")
|
||||
|
||||
// 避免重复设置相同的tab
|
||||
guard state.selectedTab != tab else {
|
||||
debugInfoSync(" ⚠️ 重复设置相同tab,忽略")
|
||||
return .none
|
||||
}
|
||||
|
||||
state.selectedTab = tab
|
||||
state.navigationPath = []
|
||||
if tab == .other, let uidStr = state.accountModel?.uid, let uid = Int(uidStr), uid > 0 {
|
||||
if state.me.uid != uid {
|
||||
state.me.uid = uid
|
||||
state.me.isFirstLoad = true // 仅当用户切换时才重置首次加载
|
||||
debugInfoSync(" ✅ selectedTab已更新为: \(state.selectedTab)")
|
||||
|
||||
// 切换到MeView时,确保有有效的uid并触发数据加载
|
||||
if tab == .other {
|
||||
if let uidStr = state.accountModel?.uid, let uid = Int(uidStr), uid > 0 {
|
||||
if state.me.displayUID != uid {
|
||||
state.me.displayUID = uid
|
||||
state.me.uid = uid // 同步更新uid
|
||||
state.me.isFirstLoad = true
|
||||
debugInfoSync(" 🔄 更新MeFeature状态,uid: \(uid)")
|
||||
}
|
||||
debugInfoSync(" 📱 切换到MeView,触发数据加载")
|
||||
return .send(.me(.onAppear))
|
||||
} else {
|
||||
debugInfoSync(" ⚠️ 切换到MeView但uid无效,等待AccountModel加载")
|
||||
}
|
||||
return .send(.me(.onAppear))
|
||||
}
|
||||
return .none
|
||||
case .feedList(.testButtonTapped):
|
||||
state.navigationPath.append(.testView)
|
||||
return .none
|
||||
case .feedList(.createFeedPublishSuccess):
|
||||
// CreateFeed发布成功,刷新FeedList和Me页数据
|
||||
return .merge(
|
||||
.send(.feedList(.reload)),
|
||||
.send(.me(.refresh))
|
||||
)
|
||||
case .feedList:
|
||||
return .none
|
||||
case let .accountModelLoaded(accountModel):
|
||||
state.accountModel = accountModel
|
||||
debugInfoSync("📦 MainFeature: AccountModel已加载")
|
||||
debugInfoSync(" uid: \(accountModel?.uid ?? "nil")")
|
||||
|
||||
// 更新MeFeature状态
|
||||
if let uidStr = accountModel?.uid, let uid = Int(uidStr), uid > 0 {
|
||||
if state.me.displayUID != uid {
|
||||
state.me.displayUID = uid
|
||||
state.me.uid = uid // 同步更新uid
|
||||
state.me.isFirstLoad = true
|
||||
debugInfoSync(" 🔄 更新MeFeature状态,uid: \(uid)")
|
||||
}
|
||||
|
||||
// 如果当前选中的是 MeView 标签页,则触发数据加载
|
||||
if state.selectedTab == .other {
|
||||
debugInfoSync(" 📱 当前在MeView,触发数据加载")
|
||||
return .send(.me(.onAppear))
|
||||
}
|
||||
|
||||
// 如果当前选中的是 FeedView 标签页,则触发数据加载
|
||||
if state.selectedTab == .feed {
|
||||
debugInfoSync(" 📱 当前在FeedView,触发数据加载")
|
||||
return .send(.feedList(.checkAuthAndLoad))
|
||||
}
|
||||
} else {
|
||||
debugInfoSync(" ⚠️ AccountModel中uid无效")
|
||||
}
|
||||
return .none
|
||||
case .me(.settingButtonTapped):
|
||||
// 触发 push 到设置页,带入当前用户信息
|
||||
@@ -96,8 +177,8 @@ struct MainFeature {
|
||||
state.appSettingState = AppSettingFeature.State(nickname: nickname, avatarURL: avatarURL, userInfo: userInfo)
|
||||
state.navigationPath.append(.appSetting)
|
||||
return .none
|
||||
case .appSettingAction(.logoutTapped):
|
||||
// 监听到登出,设置登出标志
|
||||
case .appSettingAction(.logoutConfirmed):
|
||||
// 监听到确认登出,设置登出标志
|
||||
state.isLoggedOut = true
|
||||
return .none
|
||||
case .appSettingAction(.dismissTapped):
|
||||
|
||||
@@ -14,6 +14,10 @@ struct MeDynamicFeature: Reducer {
|
||||
var hasMore: Bool = true
|
||||
var error: String?
|
||||
var isInitialized: Bool = false // 首次加载标记
|
||||
|
||||
init(uid: Int = 0) {
|
||||
self.uid = uid
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
@@ -57,7 +61,9 @@ struct MeDynamicFeature: Reducer {
|
||||
state.isLoadingMore = false
|
||||
switch result {
|
||||
case let .success(resp):
|
||||
let newDynamics = resp.data ?? []
|
||||
let myMoments = resp.data ?? []
|
||||
// 将 MyMomentInfo 转换为 MomentsInfo
|
||||
let newDynamics = myMoments.map { $0.toMomentsInfo() }
|
||||
if state.page == 1 {
|
||||
state.dynamics = newDynamics
|
||||
} else {
|
||||
@@ -76,11 +82,21 @@ struct MeDynamicFeature: Reducer {
|
||||
private func fetchDynamics(uid: Int, page: Int, pageSize: Int) -> Effect<Action> {
|
||||
let apiService = self.apiService
|
||||
return .run { send in
|
||||
debugInfoSync("🔄 MeDynamicFeature: 开始获取动态")
|
||||
debugInfoSync(" UID: \(uid)")
|
||||
debugInfoSync(" 页码: \(page)")
|
||||
debugInfoSync(" 页大小: \(pageSize)")
|
||||
|
||||
do {
|
||||
let req = GetMyDynamicRequest(fromUid: uid, uid: uid, page: page, pageSize: pageSize)
|
||||
let resp = try await apiService.request(req)
|
||||
debugInfoSync("✅ MeDynamicFeature: API 请求成功")
|
||||
debugInfoSync(" 响应码: \(resp.code)")
|
||||
debugInfoSync(" 消息: \(resp.message)")
|
||||
debugInfoSync(" 数据数量: \(resp.data?.count ?? 0)")
|
||||
await send(.fetchResponse(.success(resp)))
|
||||
} catch {
|
||||
debugErrorSync("❌ MeDynamicFeature: API 请求失败: \(error.localizedDescription)")
|
||||
await send(.fetchResponse(.failure(error as? APIError ?? .unknown(error.localizedDescription))))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -4,8 +4,10 @@ import ComposableArchitecture
|
||||
@Reducer
|
||||
struct MeFeature {
|
||||
@Dependency(\.apiService) var apiService
|
||||
@ObservableState
|
||||
struct State: Equatable {
|
||||
var isFirstLoad: Bool = true
|
||||
var isUserInfoFirstLoad: Bool = true
|
||||
var userInfo: UserInfo?
|
||||
var isLoadingUserInfo: Bool = false
|
||||
var userInfoError: String?
|
||||
@@ -18,37 +20,104 @@ struct MeFeature {
|
||||
var page: Int = 1
|
||||
var pageSize: Int = 20
|
||||
var uid: Int = 0
|
||||
// 新增:显示指定用户ID,如果为nil则显示当前登录用户
|
||||
var displayUID: Int?
|
||||
// 新增:DetailView相关状态
|
||||
var showDetail: Bool = false
|
||||
var selectedMoment: MomentsInfo?
|
||||
// 新增:错误视图相关状态
|
||||
var showErrorView: Bool = false
|
||||
var momentsFirstLoadFailed: Bool = false
|
||||
|
||||
init(displayUID: Int? = nil) {
|
||||
self.displayUID = displayUID
|
||||
// 如果displayUID不为nil,说明要显示指定用户,将其设置为uid
|
||||
if let displayUID = displayUID {
|
||||
self.uid = displayUID
|
||||
}
|
||||
}
|
||||
|
||||
// 获取实际要显示的用户ID
|
||||
var effectiveUID: Int {
|
||||
return displayUID ?? uid
|
||||
}
|
||||
|
||||
// 判断是否显示其他用户
|
||||
var isDisplayingOtherUser: Bool {
|
||||
return displayUID != nil && displayUID != uid
|
||||
}
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
case onAppear
|
||||
case refresh
|
||||
case loadMore
|
||||
case loadUserInfo
|
||||
case retryMoments
|
||||
case userInfoResponse(Result<UserInfo, APIError>)
|
||||
case momentsResponse(Result<MyMomentsResponse, APIError>)
|
||||
// 设置按钮点击
|
||||
case settingButtonTapped
|
||||
// 新增:DetailView相关Action
|
||||
case showDetail(MomentsInfo)
|
||||
case detailDismissed
|
||||
}
|
||||
|
||||
func reduce(into state: inout State, action: Action) -> Effect<Action> {
|
||||
switch action {
|
||||
case .onAppear:
|
||||
guard state.isFirstLoad else { return .none }
|
||||
state.isFirstLoad = false
|
||||
return .send(.refresh)
|
||||
debugInfoSync("\n📱 MeFeature onAppear")
|
||||
debugInfoSync(" isFirstLoad: \(state.isFirstLoad)")
|
||||
debugInfoSync(" isUserInfoFirstLoad: \(state.isUserInfoFirstLoad)")
|
||||
debugInfoSync(" effectiveUID: \(state.effectiveUID)")
|
||||
|
||||
// 每次显示都获取用户信息
|
||||
let userInfoEffect = fetchUserInfo(uid: state.effectiveUID)
|
||||
|
||||
// 只在首次进入时获取动态列表
|
||||
if state.isFirstLoad {
|
||||
state.isFirstLoad = false
|
||||
return .merge(
|
||||
userInfoEffect,
|
||||
fetchMoments(uid: state.effectiveUID, page: 1, pageSize: state.pageSize)
|
||||
)
|
||||
} else {
|
||||
return userInfoEffect
|
||||
}
|
||||
case .refresh:
|
||||
guard state.uid > 0 else { return .none }
|
||||
guard state.effectiveUID > 0 else { return .none }
|
||||
debugInfoSync("\n🔄 MeFeature refresh")
|
||||
debugInfoSync(" effectiveUID: \(state.effectiveUID)")
|
||||
state.isRefreshing = true
|
||||
state.page = 1
|
||||
state.hasMore = true
|
||||
state.userInfoError = nil // 重置错误状态
|
||||
state.momentsError = nil // 重置错误状态
|
||||
state.showErrorView = false // 隐藏错误视图
|
||||
return .merge(
|
||||
fetchUserInfo(uid: state.uid),
|
||||
fetchMoments(uid: state.uid, page: 1, pageSize: state.pageSize)
|
||||
fetchUserInfo(uid: state.effectiveUID),
|
||||
fetchMoments(uid: state.effectiveUID, page: 1, pageSize: state.pageSize)
|
||||
)
|
||||
case .loadUserInfo:
|
||||
guard state.effectiveUID > 0 else { return .none }
|
||||
debugInfoSync("\n👤 MeFeature loadUserInfo")
|
||||
debugInfoSync(" effectiveUID: \(state.effectiveUID)")
|
||||
return fetchUserInfo(uid: state.effectiveUID)
|
||||
case .retryMoments:
|
||||
guard state.effectiveUID > 0 else { return .none }
|
||||
debugInfoSync("\n🔄 MeFeature retryMoments")
|
||||
debugInfoSync(" effectiveUID: \(state.effectiveUID)")
|
||||
state.showErrorView = false // 隐藏错误视图
|
||||
state.momentsFirstLoadFailed = false
|
||||
state.isLoadingMoments = true
|
||||
state.page = 1
|
||||
state.hasMore = true
|
||||
state.momentsError = nil
|
||||
return fetchMoments(uid: state.effectiveUID, page: 1, pageSize: state.pageSize)
|
||||
case .loadMore:
|
||||
guard state.uid > 0, state.hasMore, !state.isLoadingMore else { return .none }
|
||||
guard state.effectiveUID > 0, state.hasMore, !state.isLoadingMore else { return .none }
|
||||
state.isLoadingMore = true
|
||||
return fetchMoments(uid: state.uid, page: state.page + 1, pageSize: state.pageSize)
|
||||
return fetchMoments(uid: state.effectiveUID, page: state.page + 1, pageSize: state.pageSize)
|
||||
case let .userInfoResponse(result):
|
||||
state.isLoadingUserInfo = false
|
||||
state.isRefreshing = false
|
||||
@@ -66,7 +135,49 @@ struct MeFeature {
|
||||
state.isRefreshing = false
|
||||
switch result {
|
||||
case let .success(resp):
|
||||
let newMoments = resp.data ?? []
|
||||
let myMoments = resp.data ?? []
|
||||
// 将 MyMomentInfo 转换为 MomentsInfo,并填充用户信息
|
||||
let newMoments = myMoments.map { myMoment in
|
||||
var momentsInfo = myMoment.toMomentsInfo()
|
||||
// 填充用户信息
|
||||
if let userInfo = state.userInfo {
|
||||
// 使用默认的成员初始化器
|
||||
momentsInfo = MomentsInfo(
|
||||
dynamicId: momentsInfo.dynamicId,
|
||||
uid: momentsInfo.uid,
|
||||
nick: userInfo.nick ?? userInfo.nickname ?? "未知用户",
|
||||
avatar: userInfo.avatar ?? "",
|
||||
type: momentsInfo.type,
|
||||
content: momentsInfo.content,
|
||||
likeCount: momentsInfo.likeCount,
|
||||
isLike: momentsInfo.isLike,
|
||||
commentCount: momentsInfo.commentCount,
|
||||
publishTime: momentsInfo.publishTime,
|
||||
worldId: momentsInfo.worldId,
|
||||
status: momentsInfo.status,
|
||||
playCount: momentsInfo.playCount,
|
||||
dynamicResList: momentsInfo.dynamicResList,
|
||||
gender: userInfo.gender,
|
||||
squareTop: momentsInfo.squareTop,
|
||||
topicTop: momentsInfo.topicTop,
|
||||
newUser: userInfo.newUser,
|
||||
defUser: userInfo.defUser,
|
||||
scene: momentsInfo.scene,
|
||||
userVipInfoVO: nil, // UserVipInfoVO 和 UserVipInfo 类型不匹配,暂时设为 nil
|
||||
headwearPic: userInfo.userHeadwear?.pic,
|
||||
headwearEffect: userInfo.userHeadwear?.effect,
|
||||
headwearType: userInfo.userHeadwear?.type,
|
||||
headwearName: userInfo.userHeadwear?.headwearName,
|
||||
headwearId: userInfo.userHeadwear?.headwearId,
|
||||
experLevelPic: userInfo.userLevelVo?.experUrl,
|
||||
charmLevelPic: userInfo.userLevelVo?.charmUrl,
|
||||
isCustomWord: momentsInfo.isCustomWord,
|
||||
labelList: momentsInfo.labelList
|
||||
)
|
||||
}
|
||||
return momentsInfo
|
||||
}
|
||||
|
||||
if state.page == 1 {
|
||||
state.moments = newMoments
|
||||
} else {
|
||||
@@ -75,37 +186,78 @@ struct MeFeature {
|
||||
state.hasMore = newMoments.count == state.pageSize
|
||||
if state.hasMore { state.page += 1 }
|
||||
state.momentsError = nil
|
||||
state.showErrorView = false // 隐藏错误视图
|
||||
state.momentsFirstLoadFailed = false
|
||||
|
||||
debugInfoSync("✅ 我的动态加载成功")
|
||||
debugInfoSync(" 加载数量: \(newMoments.count)")
|
||||
debugInfoSync(" 总数量: \(state.moments.count)")
|
||||
debugInfoSync(" 是否有更多: \(state.hasMore)")
|
||||
case let .failure(error):
|
||||
state.momentsError = error.localizedDescription
|
||||
// 如果是第一页加载失败,显示错误视图
|
||||
if state.page == 1 {
|
||||
state.showErrorView = true
|
||||
state.momentsFirstLoadFailed = true
|
||||
}
|
||||
debugErrorSync("❌ 我的动态加载失败: \(error.localizedDescription)")
|
||||
}
|
||||
return .none
|
||||
case .settingButtonTapped:
|
||||
// 交由 MainFeature 处理
|
||||
return .none
|
||||
case .showDetail(let moment):
|
||||
state.selectedMoment = moment
|
||||
state.showDetail = true
|
||||
return .none
|
||||
case .detailDismissed:
|
||||
state.showDetail = false
|
||||
state.selectedMoment = nil
|
||||
return .none
|
||||
}
|
||||
}
|
||||
|
||||
private func fetchUserInfo(uid: Int) -> Effect<Action> {
|
||||
.run { send in
|
||||
// do {
|
||||
if let userInfo = await UserInfoManager.fetchUserInfoFromServer(uid: String(uid), apiService: apiService) {
|
||||
await send(.userInfoResponse(.success(userInfo)))
|
||||
} else {
|
||||
await send(.userInfoResponse(.failure(.noData)))
|
||||
}
|
||||
// } catch {
|
||||
// await send(.userInfoResponse(.failure(error as? APIError ?? .unknown(error.localizedDescription))))
|
||||
// }
|
||||
debugInfoSync("👤 开始获取用户信息")
|
||||
debugInfoSync(" UID: \(uid)")
|
||||
|
||||
if let userInfo = await UserInfoManager.fetchUserInfoFromServer(uid: String(uid), apiService: apiService) {
|
||||
debugInfoSync("✅ 用户信息获取成功")
|
||||
debugInfoSync(" 昵称: \(userInfo.nick ?? userInfo.nickname ?? "未知")")
|
||||
debugInfoSync(" 头像: \(userInfo.avatar ?? "无")")
|
||||
await send(.userInfoResponse(.success(userInfo)))
|
||||
} else {
|
||||
debugErrorSync("❌ 用户信息获取失败")
|
||||
await send(.userInfoResponse(.failure(.noData)))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func fetchMoments(uid: Int, page: Int, pageSize: Int) -> Effect<Action> {
|
||||
.run { send in
|
||||
debugInfoSync("🔄 开始获取我的动态")
|
||||
debugInfoSync(" UID: \(uid)")
|
||||
debugInfoSync(" 页码: \(page)")
|
||||
debugInfoSync(" 页大小: \(pageSize)")
|
||||
|
||||
do {
|
||||
let req = GetMyDynamicRequest(fromUid: uid, uid: uid, page: page, pageSize: pageSize)
|
||||
debugInfoSync("📡 发送请求: \(req.endpoint)")
|
||||
debugInfoSync(" 参数: fromUid=\(uid), uid=\(uid), page=\(page), pageSize=\(pageSize)")
|
||||
|
||||
let resp = try await apiService.request(req)
|
||||
debugInfoSync("✅ API 请求成功")
|
||||
debugInfoSync(" 响应码: \(resp.code)")
|
||||
debugInfoSync(" 消息: \(resp.message)")
|
||||
debugInfoSync(" 数据数量: \(resp.data?.count ?? 0)")
|
||||
|
||||
await send(.momentsResponse(.success(resp)))
|
||||
} catch {
|
||||
debugErrorSync("❌ API 请求失败: \(error.localizedDescription)")
|
||||
if let apiError = error as? APIError {
|
||||
debugErrorSync(" API错误类型: \(apiError)")
|
||||
}
|
||||
await send(.momentsResponse(.failure(error as? APIError ?? .unknown(error.localizedDescription))))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -57,12 +57,12 @@ struct RecoverPasswordFeature {
|
||||
|
||||
case .getVerificationCodeTapped:
|
||||
guard !state.email.isEmpty else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.email_required", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.email_required", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
guard ValidationHelper.isValidEmail(state.email) else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.invalid_email", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.invalid_email", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
@@ -101,23 +101,23 @@ struct RecoverPasswordFeature {
|
||||
if let apiError = error as? APIError {
|
||||
state.errorMessage = apiError.localizedDescription
|
||||
} else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.code_send_failed", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.code_send_failed", comment: "")
|
||||
}
|
||||
return .none
|
||||
|
||||
case .resetPasswordTapped:
|
||||
guard !state.email.isEmpty && !state.verificationCode.isEmpty && !state.newPassword.isEmpty else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.fields_required", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.fields_required", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
guard ValidationHelper.isValidEmail(state.email) else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.invalid_email", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.invalid_email", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
guard ValidationHelper.isValidPassword(state.newPassword) else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.invalid_password", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.invalid_password", comment: "")
|
||||
return .none
|
||||
}
|
||||
|
||||
@@ -160,7 +160,7 @@ struct RecoverPasswordFeature {
|
||||
if let apiError = error as? APIError {
|
||||
state.errorMessage = apiError.localizedDescription
|
||||
} else {
|
||||
state.errorMessage = NSLocalizedString("recover_password.reset_failed", comment: "")
|
||||
state.errorMessage = LocalizedStringSync("recover_password.reset_failed", comment: "")
|
||||
}
|
||||
return .none
|
||||
|
||||
@@ -199,7 +199,7 @@ struct ResetPasswordResponse: Codable, Equatable {
|
||||
|
||||
/// 错误消息(如果有)
|
||||
var errorMessage: String {
|
||||
return message ?? NSLocalizedString("recover_password.reset_failed", comment: "")
|
||||
return message ?? LocalizedStringSync("recover_password.reset_failed", comment: "")
|
||||
}
|
||||
}
|
||||
|
||||
@@ -210,21 +210,32 @@ struct ResetPasswordRequest: APIRequestProtocol {
|
||||
let endpoint = "/acc/pwd/resetByEmail" // 新的API端点
|
||||
let method: HTTPMethod = .POST
|
||||
let includeBaseParameters = true
|
||||
let queryParameters: [String: String]?
|
||||
var bodyParameters: [String: Any]? { nil }
|
||||
let timeout: TimeInterval = 30.0
|
||||
|
||||
// MARK: - Private Properties
|
||||
private let email: String
|
||||
private let code: String
|
||||
private let newPwd: String
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var queryParameters: [String: String]? {
|
||||
return [
|
||||
"email": email,
|
||||
"newPwd": newPwd, // 参数名改为newPwd
|
||||
"code": code
|
||||
]
|
||||
}
|
||||
|
||||
/// 初始化密码重置请求
|
||||
/// - Parameters:
|
||||
/// - email: DES加密后的邮箱地址
|
||||
/// - code: 验证码
|
||||
/// - newPwd: DES加密后的新密码
|
||||
init(email: String, code: String, newPwd: String) {
|
||||
self.queryParameters = [
|
||||
"email": email,
|
||||
"newPwd": newPwd, // 参数名改为newPwd
|
||||
"code": code
|
||||
]
|
||||
self.email = email
|
||||
self.code = code
|
||||
self.newPwd = newPwd
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,110 +0,0 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
|
||||
@Reducer
|
||||
struct SplashFeature {
|
||||
@ObservableState
|
||||
struct State: Equatable {
|
||||
var isLoading = true
|
||||
var shouldShowMainApp = false
|
||||
var authenticationStatus: UserInfoManager.AuthenticationStatus = .notFound
|
||||
var isCheckingAuthentication = false
|
||||
|
||||
// 新增:导航目标
|
||||
var navigationDestination: NavigationDestination?
|
||||
}
|
||||
|
||||
// 新增:导航目标枚举
|
||||
enum NavigationDestination: Equatable {
|
||||
case login // 跳转到登录页面
|
||||
case main // 跳转到主页面
|
||||
}
|
||||
|
||||
enum Action: Equatable {
|
||||
case onAppear
|
||||
case splashFinished
|
||||
case checkAuthentication
|
||||
case authenticationChecked(UserInfoManager.AuthenticationStatus)
|
||||
|
||||
// 新增:用户信息获取 actions
|
||||
case fetchUserInfo
|
||||
case userInfoFetched(Bool)
|
||||
|
||||
// 新增:导航 actions
|
||||
case navigateToLogin
|
||||
case navigateToMain
|
||||
}
|
||||
|
||||
@Dependency(\.apiService) var apiService // 新增:API服务依赖
|
||||
|
||||
var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .onAppear:
|
||||
state.isLoading = true
|
||||
state.shouldShowMainApp = false
|
||||
state.authenticationStatus = UserInfoManager.AuthenticationStatus.notFound
|
||||
state.isCheckingAuthentication = false
|
||||
state.navigationDestination = nil
|
||||
|
||||
// 1秒延迟后显示主应用 (iOS 15.5+ 兼容)
|
||||
return .run { send in
|
||||
try await Task.sleep(nanoseconds: 1_000_000_000) // 1秒 = 1,000,000,000 纳秒
|
||||
await send(.splashFinished)
|
||||
}
|
||||
case .splashFinished:
|
||||
state.isLoading = false
|
||||
|
||||
// Splash 完成后,开始检查认证状态
|
||||
return .send(.checkAuthentication)
|
||||
|
||||
case .checkAuthentication:
|
||||
state.isCheckingAuthentication = true
|
||||
|
||||
// 异步检查认证状态
|
||||
return .run { send in
|
||||
let authStatus = await UserInfoManager.checkAuthenticationStatus()
|
||||
await send(.authenticationChecked(authStatus))
|
||||
}
|
||||
|
||||
case let .authenticationChecked(status):
|
||||
state.isCheckingAuthentication = false
|
||||
state.authenticationStatus = status
|
||||
|
||||
// 根据认证状态决定下一步操作
|
||||
if status.canAutoLogin {
|
||||
debugInfoSync("🎉 自动登录成功,开始获取用户信息")
|
||||
// 新增:认证成功后自动获取用户信息
|
||||
return .send(.fetchUserInfo)
|
||||
} else {
|
||||
debugInfoSync("🔑 需要手动登录")
|
||||
return .send(.navigateToLogin)
|
||||
}
|
||||
|
||||
case .fetchUserInfo:
|
||||
// 新增:获取用户信息
|
||||
return .run { send in
|
||||
let success = await UserInfoManager.autoFetchUserInfoOnAppLaunch(apiService: apiService)
|
||||
await send(.userInfoFetched(success))
|
||||
}
|
||||
|
||||
case let .userInfoFetched(success):
|
||||
if success {
|
||||
debugInfoSync("✅ 用户信息获取成功,进入主页")
|
||||
} else {
|
||||
debugInfoSync("⚠️ 用户信息获取失败,但仍进入主页")
|
||||
}
|
||||
return .send(.navigateToMain)
|
||||
|
||||
case .navigateToLogin:
|
||||
state.navigationDestination = .login
|
||||
return .none
|
||||
|
||||
case .navigateToMain:
|
||||
state.navigationDestination = .main
|
||||
state.shouldShowMainApp = true
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -15,5 +15,7 @@
|
||||
<array>
|
||||
<string>Bayon-Regular.ttf</string>
|
||||
</array>
|
||||
<key>API_SIGNING_KEY</key>
|
||||
<string></string>
|
||||
</dict>
|
||||
</plist>
|
||||
|
||||
@@ -1,63 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="23727" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES">
|
||||
<device id="retina6_12" orientation="portrait" appearance="light"/>
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="23721"/>
|
||||
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
|
||||
<capability name="System colors in document resources" minToolsVersion="11.0"/>
|
||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--View Controller-->
|
||||
<scene sceneID="s0d-6b-0kx">
|
||||
<objects>
|
||||
<viewController id="Y6W-OH-hqX" sceneMemberID="viewController">
|
||||
<view key="view" contentMode="scaleToFill" id="5EZ-qb-Rvc">
|
||||
<rect key="frame" x="0.0" y="0.0" width="393" height="852"/>
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<subviews>
|
||||
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFit" horizontalHuggingPriority="251" verticalHuggingPriority="251" image="bg" translatesAutoresizingMaskIntoConstraints="NO" id="Mom-Je-A43">
|
||||
<rect key="frame" x="0.0" y="0.0" width="393" height="852"/>
|
||||
</imageView>
|
||||
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFit" horizontalHuggingPriority="251" verticalHuggingPriority="251" image="logo" translatesAutoresizingMaskIntoConstraints="NO" id="jVZ-ey-zjS">
|
||||
<rect key="frame" x="146.66666666666666" y="200" width="100" height="100"/>
|
||||
<constraints>
|
||||
<constraint firstAttribute="width" constant="100" id="61A-Xv-MlD"/>
|
||||
<constraint firstAttribute="height" constant="100" id="NWJ-mJ-K2O"/>
|
||||
</constraints>
|
||||
</imageView>
|
||||
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="E-Parti" textAlignment="center" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="GrU-nK-tAY">
|
||||
<rect key="frame" x="138" y="332" width="117" height="48"/>
|
||||
<fontDescription key="fontDescription" type="system" pointSize="40"/>
|
||||
<color key="textColor" white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/>
|
||||
<nil key="highlightedColor"/>
|
||||
</label>
|
||||
</subviews>
|
||||
<viewLayoutGuide key="safeArea" id="vDu-zF-Fre"/>
|
||||
<color key="backgroundColor" systemColor="systemBackgroundColor"/>
|
||||
<constraints>
|
||||
<constraint firstItem="jVZ-ey-zjS" firstAttribute="centerX" secondItem="Mom-Je-A43" secondAttribute="centerX" id="Atv-LB-aNW"/>
|
||||
<constraint firstItem="Mom-Je-A43" firstAttribute="leading" secondItem="vDu-zF-Fre" secondAttribute="leading" id="fQ2-yj-nze"/>
|
||||
<constraint firstItem="Mom-Je-A43" firstAttribute="top" secondItem="5EZ-qb-Rvc" secondAttribute="top" id="gVA-6N-OG4"/>
|
||||
<constraint firstItem="GrU-nK-tAY" firstAttribute="centerY" secondItem="Mom-Je-A43" secondAttribute="centerY" constant="-70" id="j81-qa-9vS"/>
|
||||
<constraint firstAttribute="bottom" secondItem="Mom-Je-A43" secondAttribute="bottom" id="lBn-me-aJu"/>
|
||||
<constraint firstItem="Mom-Je-A43" firstAttribute="trailing" secondItem="vDu-zF-Fre" secondAttribute="trailing" id="lka-KN-fEy"/>
|
||||
<constraint firstItem="jVZ-ey-zjS" firstAttribute="top" secondItem="Mom-Je-A43" secondAttribute="top" constant="200" id="nCq-oK-mTB"/>
|
||||
<constraint firstItem="GrU-nK-tAY" firstAttribute="centerX" secondItem="Mom-Je-A43" secondAttribute="centerX" id="sZE-bZ-0Xj"/>
|
||||
</constraints>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="Ief-a0-LHa" userLabel="First Responder" customClass="UIResponder" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
<point key="canvasLocation" x="-208.3969465648855" y="-13.380281690140846"/>
|
||||
</scene>
|
||||
</scenes>
|
||||
<resources>
|
||||
<image name="bg" width="375" height="812"/>
|
||||
<image name="logo" width="100" height="100"/>
|
||||
<systemColor name="systemBackgroundColor">
|
||||
<color white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/>
|
||||
</systemColor>
|
||||
</resources>
|
||||
</document>
|
||||
429
yana/MVVM/CommonComponents.swift
Normal file
429
yana/MVVM/CommonComponents.swift
Normal file
@@ -0,0 +1,429 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - App Image Source Enum
|
||||
enum AppImageSource: Equatable {
|
||||
case camera
|
||||
case photoLibrary
|
||||
}
|
||||
|
||||
// MARK: - 通用底部 Tab 栏组件
|
||||
public struct TabBarItem: Identifiable, Equatable {
|
||||
public let id: String
|
||||
public let title: String
|
||||
public let systemIconName: String
|
||||
public init(id: String, title: String, systemIconName: String) {
|
||||
self.id = id
|
||||
self.title = title
|
||||
self.systemIconName = systemIconName
|
||||
}
|
||||
}
|
||||
|
||||
struct BottomTabBar: View {
|
||||
let items: [TabBarItem]
|
||||
@Binding var selectedId: String
|
||||
let onSelect: (String) -> Void
|
||||
var contentPadding: EdgeInsets = EdgeInsets(top: 12, leading: 0, bottom: 12, trailing: 0)
|
||||
var horizontalPadding: CGFloat = 0
|
||||
|
||||
// 便捷初始化:内部固定 tabs,避免外部重复声明
|
||||
init(
|
||||
selectedId: Binding<String>,
|
||||
onSelect: @escaping (String) -> Void,
|
||||
contentPadding: EdgeInsets = EdgeInsets(top: 12, leading: 0, bottom: 12, trailing: 0),
|
||||
horizontalPadding: CGFloat = 0
|
||||
) {
|
||||
self.items = BottomTabBar.defaultItems()
|
||||
self._selectedId = selectedId
|
||||
self.onSelect = onSelect
|
||||
self.contentPadding = contentPadding
|
||||
self.horizontalPadding = horizontalPadding
|
||||
}
|
||||
|
||||
// 最简初始化:直接接受 viewModel,内部处理所有逻辑
|
||||
init(viewModel: MainViewModel) {
|
||||
self.items = BottomTabBar.defaultItems()
|
||||
self._selectedId = Binding(
|
||||
get: { viewModel.selectedTab.rawValue },
|
||||
set: { raw in
|
||||
if let tab = MainViewModel.Tab(rawValue: raw) {
|
||||
viewModel.onTabChanged(tab)
|
||||
}
|
||||
}
|
||||
)
|
||||
self.onSelect = { _ in } // 保留但不再使用
|
||||
self.contentPadding = EdgeInsets(top: 12, leading: 0, bottom: 12, trailing: 0)
|
||||
self.horizontalPadding = 0
|
||||
}
|
||||
|
||||
// 使用 BottomTabView.swift 中的图片资源名进行映射
|
||||
private func assetIconName(for item: TabBarItem, isSelected: Bool) -> String? {
|
||||
switch item.id {
|
||||
case "feed":
|
||||
return isSelected ? "feed selected" : "feed unselected"
|
||||
case "me":
|
||||
return isSelected ? "me selected" : "me unselected"
|
||||
default:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
// 内部默认 items(与资源映射保持一致)
|
||||
private static func defaultItems() -> [TabBarItem] {
|
||||
return [
|
||||
TabBarItem(id: "feed", title: "Feed", systemIconName: "list.bullet"),
|
||||
TabBarItem(id: "me", title: "Me", systemIconName: "person.circle")
|
||||
]
|
||||
}
|
||||
|
||||
var body: some View {
|
||||
HStack(spacing: 8) {
|
||||
ForEach(items) { item in
|
||||
Button(action: {
|
||||
selectedId = item.id
|
||||
onSelect(item.id)
|
||||
}) {
|
||||
Group {
|
||||
if let name = assetIconName(for: item, isSelected: selectedId == item.id) {
|
||||
Image(name)
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fit)
|
||||
.frame(width: 30, height: 30)
|
||||
} else {
|
||||
Image(systemName: item.systemIconName)
|
||||
.font(.system(size: 24))
|
||||
.foregroundColor(selectedId == item.id ? .white : .white.opacity(0.6))
|
||||
}
|
||||
}
|
||||
}
|
||||
.frame(maxWidth: .infinity)
|
||||
.padding(contentPadding)
|
||||
.contentShape(Rectangle())
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, 8) // 按钮与边缘保持 8 间距
|
||||
.padding(.horizontal, horizontalPadding)
|
||||
.background(LiquidGlassBackground())
|
||||
.clipShape(Capsule())
|
||||
.contentShape(Capsule())
|
||||
.onTapGesture { /* 吸收空白区域点击,避免穿透 */ }
|
||||
.overlay(
|
||||
Capsule()
|
||||
.stroke(Color.white.opacity(0.12), lineWidth: 0.5)
|
||||
)
|
||||
.shadow(color: Color.black.opacity(0.2), radius: 12, x: 0, y: 6)
|
||||
.safeAreaInset(edge: .bottom) {
|
||||
Color.clear.frame(height: 0)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Liquid Glass Background (iOS 26 优先,向下优雅降级)
|
||||
struct LiquidGlassBackground: View {
|
||||
var body: some View {
|
||||
Group {
|
||||
if #available(iOS 26.0, *) {
|
||||
// iOS 26+:使用系统液态玻璃效果
|
||||
Rectangle()
|
||||
.fill(Color.clear)
|
||||
.glassEffect()
|
||||
} else
|
||||
if #available(iOS 17.0, *) {
|
||||
// iOS 17-25:使用超薄材质 + 轻微高光层
|
||||
ZStack {
|
||||
Rectangle().fill(.ultraThinMaterial)
|
||||
LinearGradient(
|
||||
colors: [Color.white.opacity(0.06), Color.clear, Color.white.opacity(0.04)],
|
||||
startPoint: .topLeading,
|
||||
endPoint: .bottomTrailing
|
||||
)
|
||||
.blendMode(.softLight)
|
||||
}
|
||||
} else {
|
||||
// 更低版本:半透明备选
|
||||
Rectangle()
|
||||
.fill(Color.black.opacity(0.2))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 背景视图组件
|
||||
struct LoginBackgroundView: View {
|
||||
var body: some View {
|
||||
Image("bg")
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
// .ignoresSafeArea(.all)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 顶部导航栏组件
|
||||
struct LoginHeaderView: View {
|
||||
let onBack: () -> Void
|
||||
|
||||
var body: some View {
|
||||
HStack {
|
||||
Button(action: onBack) {
|
||||
Image(systemName: "chevron.left")
|
||||
.font(.system(size: 24, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.frame(width: 44, height: 44)
|
||||
}
|
||||
Spacer()
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.top, 8)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 通用输入框组件
|
||||
enum InputFieldType {
|
||||
case text
|
||||
case number
|
||||
case password
|
||||
case verificationCode
|
||||
}
|
||||
|
||||
struct CustomInputField: View {
|
||||
let type: InputFieldType
|
||||
let placeholder: String
|
||||
let text: Binding<String>
|
||||
let isPasswordVisible: Binding<Bool>?
|
||||
let onGetCode: (() -> Void)?
|
||||
let isCodeButtonEnabled: Bool
|
||||
let isCodeLoading: Bool
|
||||
let getCodeButtonText: String
|
||||
|
||||
init(
|
||||
type: InputFieldType,
|
||||
placeholder: String,
|
||||
text: Binding<String>,
|
||||
isPasswordVisible: Binding<Bool>? = nil,
|
||||
onGetCode: (() -> Void)? = nil,
|
||||
isCodeButtonEnabled: Bool = false,
|
||||
isCodeLoading: Bool = false,
|
||||
getCodeButtonText: String = ""
|
||||
) {
|
||||
self.type = type
|
||||
self.placeholder = placeholder
|
||||
self.text = text
|
||||
self.isPasswordVisible = isPasswordVisible
|
||||
self.onGetCode = onGetCode
|
||||
self.isCodeButtonEnabled = isCodeButtonEnabled
|
||||
self.isCodeLoading = isCodeLoading
|
||||
self.getCodeButtonText = getCodeButtonText
|
||||
}
|
||||
|
||||
var body: some View {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.fill(Color.white.opacity(0.1))
|
||||
.overlay(
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.stroke(Color.white.opacity(0.3), lineWidth: 1)
|
||||
)
|
||||
.frame(height: 56)
|
||||
|
||||
HStack {
|
||||
// 输入框
|
||||
Group {
|
||||
switch type {
|
||||
case .text, .number:
|
||||
TextField("", text: text)
|
||||
.placeholder(when: text.wrappedValue.isEmpty) {
|
||||
Text(placeholder)
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.keyboardType(type == .number ? .numberPad : .default)
|
||||
case .password:
|
||||
if let isPasswordVisible = isPasswordVisible {
|
||||
if isPasswordVisible.wrappedValue {
|
||||
TextField("", text: text)
|
||||
.placeholder(when: text.wrappedValue.isEmpty) {
|
||||
Text(placeholder)
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
} else {
|
||||
SecureField("", text: text)
|
||||
.placeholder(when: text.wrappedValue.isEmpty) {
|
||||
Text(placeholder)
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
}
|
||||
}
|
||||
case .verificationCode:
|
||||
TextField("", text: text)
|
||||
.placeholder(when: text.wrappedValue.isEmpty) {
|
||||
Text(placeholder)
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.keyboardType(.numberPad)
|
||||
}
|
||||
}
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 16))
|
||||
|
||||
// 右侧按钮
|
||||
if type == .password, let isPasswordVisible = isPasswordVisible {
|
||||
Button(action: {
|
||||
isPasswordVisible.wrappedValue.toggle()
|
||||
}) {
|
||||
Image(systemName: isPasswordVisible.wrappedValue ? "eye.slash" : "eye")
|
||||
.foregroundColor(.white.opacity(0.7))
|
||||
.font(.system(size: 18))
|
||||
}
|
||||
} else if type == .verificationCode, let onGetCode = onGetCode {
|
||||
Button(action: onGetCode) {
|
||||
ZStack {
|
||||
if isCodeLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.7)
|
||||
} else {
|
||||
Text(getCodeButtonText)
|
||||
.font(.system(size: 14, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
}
|
||||
}
|
||||
.frame(width: 60, height: 36)
|
||||
.background(
|
||||
RoundedRectangle(cornerRadius: 15)
|
||||
.fill(Color.white.opacity(isCodeButtonEnabled ? 0.2 : 0.1))
|
||||
)
|
||||
}
|
||||
.disabled(!isCodeButtonEnabled || isCodeLoading)
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, 24)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 登录按钮组件
|
||||
struct LoginButtonView: View {
|
||||
let isLoading: Bool
|
||||
let isEnabled: Bool
|
||||
let onTap: () -> Void
|
||||
|
||||
var body: some View {
|
||||
Button(action: onTap) {
|
||||
Group {
|
||||
if isLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(1.2)
|
||||
} else {
|
||||
Text("Login")
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
}
|
||||
}
|
||||
}
|
||||
.frame(maxWidth: .infinity)
|
||||
.padding(.vertical, 16)
|
||||
.background(isEnabled ? Color(red: 0.5, green: 0.3, blue: 0.8) : Color.gray)
|
||||
.cornerRadius(8)
|
||||
.disabled(!isEnabled)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 设置行组件
|
||||
struct SettingRow: View {
|
||||
let title: String
|
||||
let subtitle: String
|
||||
let action: (() -> Void)?
|
||||
|
||||
var body: some View {
|
||||
Button(action: {
|
||||
action?()
|
||||
}) {
|
||||
HStack(spacing: 16) {
|
||||
HStack {
|
||||
Text(title)
|
||||
.font(.system(size: 16))
|
||||
.foregroundColor(.white)
|
||||
.multilineTextAlignment(.leading)
|
||||
|
||||
Spacer()
|
||||
|
||||
if !subtitle.isEmpty {
|
||||
Text(subtitle)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.7))
|
||||
}
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
if action != nil {
|
||||
Image(systemName: "chevron.right")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.5))
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.vertical, 12)
|
||||
}
|
||||
.disabled(action == nil)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Camera Picker
|
||||
struct CameraPicker: UIViewControllerRepresentable {
|
||||
let onImagePicked: (UIImage?) -> Void
|
||||
|
||||
func makeUIViewController(context: Context) -> UIImagePickerController {
|
||||
let picker = UIImagePickerController()
|
||||
picker.sourceType = .camera
|
||||
picker.delegate = context.coordinator
|
||||
return picker
|
||||
}
|
||||
|
||||
func updateUIViewController(_ uiViewController: UIImagePickerController, context: Context) {}
|
||||
|
||||
func makeCoordinator() -> Coordinator {
|
||||
Coordinator(onImagePicked: onImagePicked)
|
||||
}
|
||||
|
||||
class Coordinator: NSObject, UIImagePickerControllerDelegate, UINavigationControllerDelegate {
|
||||
let onImagePicked: (UIImage?) -> Void
|
||||
|
||||
init(onImagePicked: @escaping (UIImage?) -> Void) {
|
||||
self.onImagePicked = onImagePicked
|
||||
}
|
||||
|
||||
func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) {
|
||||
if let image = info[.originalImage] as? UIImage {
|
||||
onImagePicked(image)
|
||||
} else {
|
||||
onImagePicked(nil)
|
||||
}
|
||||
picker.dismiss(animated: true)
|
||||
}
|
||||
|
||||
func imagePickerControllerDidCancel(_ picker: UIImagePickerController) {
|
||||
onImagePicked(nil)
|
||||
picker.dismiss(animated: true)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#Preview {
|
||||
VStack(spacing: 20) {
|
||||
LoginBackgroundView()
|
||||
|
||||
LoginHeaderView(onBack: {})
|
||||
|
||||
CustomInputField(
|
||||
type: .text,
|
||||
placeholder: "Test Input",
|
||||
text: .constant("")
|
||||
)
|
||||
|
||||
LoginButtonView(
|
||||
isLoading: false,
|
||||
isEnabled: true,
|
||||
onTap: {}
|
||||
)
|
||||
}
|
||||
}
|
||||
230
yana/MVVM/CreateFeedPage.swift
Normal file
230
yana/MVVM/CreateFeedPage.swift
Normal file
@@ -0,0 +1,230 @@
|
||||
import SwiftUI
|
||||
import PhotosUI
|
||||
|
||||
@MainActor
|
||||
final class CreateFeedViewModel: ObservableObject {
|
||||
@Published var content: String = ""
|
||||
@Published var selectedImages: [UIImage] = []
|
||||
@Published var isPublishing: Bool = false
|
||||
@Published var errorMessage: String? = nil
|
||||
// 仅当有文本时才允许发布
|
||||
var canPublish: Bool { !content.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty }
|
||||
}
|
||||
|
||||
struct CreateFeedPage: View {
|
||||
@Environment(\.dismiss) private var dismiss
|
||||
@StateObject private var viewModel = CreateFeedViewModel()
|
||||
let onDismiss: () -> Void
|
||||
|
||||
// MARK: - UI State
|
||||
@FocusState private var isTextEditorFocused: Bool
|
||||
@State private var isShowingPreview: Bool = false
|
||||
@State private var previewIndex: Int = 0
|
||||
|
||||
private let maxCharacters: Int = 500
|
||||
private let gridSpacing: CGFloat = 8
|
||||
private let gridCornerRadius: CGFloat = 16
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
Color(hex: 0x0C0527)
|
||||
.ignoresSafeArea()
|
||||
.onTapGesture {
|
||||
// 点击背景收起键盘
|
||||
isTextEditorFocused = false
|
||||
}
|
||||
VStack(spacing: 16) {
|
||||
HStack {
|
||||
Button(action: {
|
||||
onDismiss()
|
||||
dismiss()
|
||||
}) {
|
||||
Image(systemName: "xmark")
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 18, weight: .medium))
|
||||
.frame(width: 44, height: 44, alignment: .center)
|
||||
.contentShape(Rectangle())
|
||||
}
|
||||
Spacer()
|
||||
Text(LocalizedString ("createFeed.title", comment: "Image & Text Publish"))
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 18, weight: .medium))
|
||||
Spacer()
|
||||
Button(action: publish) {
|
||||
if viewModel.isPublishing {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
} else {
|
||||
Text(LocalizedString("createFeed.publish", comment: "Publish"))
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 14, weight: .medium))
|
||||
}
|
||||
}
|
||||
.disabled(!viewModel.canPublish || viewModel.isPublishing)
|
||||
.opacity((!viewModel.canPublish || viewModel.isPublishing) ? 0.6 : 1)
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.top, 12)
|
||||
.contentShape(Rectangle())
|
||||
.zIndex(10)
|
||||
|
||||
ZStack(alignment: .topLeading) {
|
||||
RoundedRectangle(cornerRadius: 8).fill(Color(hex: 0x1C143A))
|
||||
if viewModel.content.isEmpty {
|
||||
Text(LocalizedString("createFeed.enterContent", comment: "Enter Content"))
|
||||
.foregroundColor(.white.opacity(0.5))
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.vertical, 12)
|
||||
}
|
||||
TextEditor(text: $viewModel.content)
|
||||
.foregroundColor(.white)
|
||||
.padding(.horizontal, 12)
|
||||
.padding(.vertical, 8)
|
||||
.scrollContentBackground(.hidden)
|
||||
.focused($isTextEditorFocused)
|
||||
.frame(height: 200)
|
||||
.zIndex(1) // 确保编辑器不会遮挡顶部栏的点击
|
||||
|
||||
// 字数统计(右下角)
|
||||
VStack { Spacer() }
|
||||
.overlay(alignment: .bottomTrailing) {
|
||||
Text("\(viewModel.content.count)/\(maxCharacters)")
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
.font(.system(size: 14))
|
||||
.padding(.trailing, 8)
|
||||
.padding(.bottom, 8)
|
||||
}
|
||||
}
|
||||
.frame(height: 200)
|
||||
.padding(.horizontal, 20)
|
||||
.onChange(of: viewModel.content) { _, newValue in
|
||||
// 限制最大字数
|
||||
if newValue.count > maxCharacters {
|
||||
viewModel.content = String(newValue.prefix(maxCharacters))
|
||||
}
|
||||
}
|
||||
|
||||
NineGridImagePicker(
|
||||
images: $viewModel.selectedImages,
|
||||
maxCount: 9,
|
||||
cornerRadius: gridCornerRadius,
|
||||
spacing: gridSpacing,
|
||||
horizontalPadding: 20,
|
||||
onTapImage: { index in
|
||||
previewIndex = index
|
||||
isShowingPreview = true
|
||||
}
|
||||
)
|
||||
|
||||
if let error = viewModel.errorMessage {
|
||||
Text(error)
|
||||
.foregroundColor(.red)
|
||||
.font(.system(size: 14))
|
||||
}
|
||||
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
}
|
||||
.navigationBarBackButtonHidden(true)
|
||||
.fullScreenCover(isPresented: $isShowingPreview) {
|
||||
ZStack {
|
||||
Color.black.ignoresSafeArea()
|
||||
VStack(spacing: 0) {
|
||||
HStack {
|
||||
Spacer()
|
||||
Button {
|
||||
isShowingPreview = false
|
||||
} label: {
|
||||
Image(systemName: "xmark")
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 18, weight: .medium))
|
||||
.padding(12)
|
||||
}
|
||||
}
|
||||
.padding(.top, 8)
|
||||
|
||||
TabView(selection: $previewIndex) {
|
||||
ForEach(viewModel.selectedImages.indices, id: \.self) { idx in
|
||||
ZStack {
|
||||
Color.black
|
||||
Image(uiImage: viewModel.selectedImages[idx])
|
||||
.resizable()
|
||||
.scaledToFit()
|
||||
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||
}
|
||||
.tag(idx)
|
||||
}
|
||||
}
|
||||
.tabViewStyle(.page(indexDisplayMode: .automatic))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func publish() {
|
||||
viewModel.isPublishing = true
|
||||
viewModel.errorMessage = nil
|
||||
Task { @MainActor in
|
||||
let apiService: any APIServiceProtocol & Sendable = LiveAPIService()
|
||||
do {
|
||||
// 1) 上传图片(如有)
|
||||
var resList: [ResListItem] = []
|
||||
if !viewModel.selectedImages.isEmpty {
|
||||
for image in viewModel.selectedImages {
|
||||
if let url = await COSManager.shared.uploadUIImage(image, apiService: apiService) {
|
||||
if let cg = image.cgImage {
|
||||
let item = ResListItem(resUrl: url, width: cg.width, height: cg.height, format: "jpeg")
|
||||
resList.append(item)
|
||||
} else {
|
||||
// 无法获取尺寸也允许发布,尺寸置为 0
|
||||
let item = ResListItem(resUrl: url, width: 0, height: 0, format: "jpeg")
|
||||
resList.append(item)
|
||||
}
|
||||
} else {
|
||||
viewModel.isPublishing = false
|
||||
viewModel.errorMessage = "图片上传失败"
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 2) 组装并发送发布请求
|
||||
let trimmed = viewModel.content.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
let userId = await UserInfoManager.getCurrentUserId() ?? ""
|
||||
let type = resList.isEmpty ? "0" : "2" // 0: 纯文字, 2: 图片/图文
|
||||
let request = await PublishFeedRequest.make(
|
||||
content: trimmed,
|
||||
uid: userId,
|
||||
type: type,
|
||||
resList: resList.isEmpty ? nil : resList
|
||||
)
|
||||
let response = try await apiService.request(request)
|
||||
|
||||
// 3) 结果处理
|
||||
if response.code == 200 {
|
||||
viewModel.isPublishing = false
|
||||
NotificationCenter.default.post(name: .init("CreateFeedPublished"), object: nil)
|
||||
onDismiss()
|
||||
dismiss()
|
||||
} else {
|
||||
viewModel.isPublishing = false
|
||||
viewModel.errorMessage = response.message.isEmpty ? "发布失败" : response.message
|
||||
}
|
||||
} catch {
|
||||
viewModel.isPublishing = false
|
||||
viewModel.errorMessage = error.localizedDescription
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func removeImage(at index: Int) {
|
||||
guard viewModel.selectedImages.indices.contains(index) else { return }
|
||||
viewModel.selectedImages.remove(at: index)
|
||||
if isShowingPreview {
|
||||
if previewIndex >= viewModel.selectedImages.count { previewIndex = max(0, viewModel.selectedImages.count - 1) }
|
||||
if viewModel.selectedImages.isEmpty { isShowingPreview = false }
|
||||
}
|
||||
}
|
||||
}
|
||||
332
yana/MVVM/EMailLoginPage.swift
Normal file
332
yana/MVVM/EMailLoginPage.swift
Normal file
@@ -0,0 +1,332 @@
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
// MARK: - EMailLogin ViewModel
|
||||
|
||||
@MainActor
|
||||
class EMailLoginViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var email: String = ""
|
||||
@Published var verificationCode: String = ""
|
||||
@Published var codeCountdown: Int = 0
|
||||
@Published var isLoading: Bool = false
|
||||
@Published var isCodeLoading: Bool = false
|
||||
@Published var errorMessage: String?
|
||||
@Published var loginStep: LoginStep = .input
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onBack: (() -> Void)?
|
||||
var onLoginSuccess: (() -> Void)?
|
||||
|
||||
// MARK: - Private Properties
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
private var timerCancellable: AnyCancellable?
|
||||
|
||||
// MARK: - Enums
|
||||
enum LoginStep: Equatable {
|
||||
case input
|
||||
case completed
|
||||
}
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var isLoginButtonEnabled: Bool {
|
||||
return !isLoading && !email.isEmpty && !verificationCode.isEmpty
|
||||
}
|
||||
|
||||
var getCodeButtonText: String {
|
||||
if codeCountdown > 0 {
|
||||
return "\(codeCountdown)s"
|
||||
} else {
|
||||
return "Get"
|
||||
}
|
||||
}
|
||||
|
||||
var isCodeButtonEnabled: Bool {
|
||||
return !isCodeLoading && codeCountdown == 0 && !email.isEmpty
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onBackTapped() {
|
||||
onBack?()
|
||||
}
|
||||
|
||||
func onEmailChanged(_ newEmail: String) {
|
||||
email = newEmail
|
||||
}
|
||||
|
||||
func onVerificationCodeChanged(_ newCode: String) {
|
||||
verificationCode = newCode
|
||||
}
|
||||
|
||||
func onGetVerificationCodeTapped() {
|
||||
guard isCodeButtonEnabled else { return }
|
||||
|
||||
isCodeLoading = true
|
||||
errorMessage = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let result = try await requestVerificationCode()
|
||||
await MainActor.run {
|
||||
self.handleCodeRequestResult(result)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleCodeRequestError(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func onLoginTapped() {
|
||||
guard isLoginButtonEnabled else { return }
|
||||
|
||||
isLoading = true
|
||||
errorMessage = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let result = try await performLogin()
|
||||
await MainActor.run {
|
||||
self.handleLoginResult(result)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleLoginError(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func resetState() {
|
||||
email = ""
|
||||
verificationCode = ""
|
||||
codeCountdown = 0
|
||||
isLoading = false
|
||||
isCodeLoading = false
|
||||
errorMessage = nil
|
||||
loginStep = .input
|
||||
stopCountdown()
|
||||
}
|
||||
|
||||
// MARK: - Private Methods
|
||||
private func requestVerificationCode() async throws -> Bool {
|
||||
return false
|
||||
// let request = EmailVerificationCodeRequest(email: email)
|
||||
// let apiService = LiveAPIService()
|
||||
// let response: EmailVerificationCodeResponse = try await apiService.request(request)
|
||||
//
|
||||
// if response.code == 200 {
|
||||
// return true
|
||||
// } else {
|
||||
// throw APIError.serverError(response.message ?? "Failed to send verification code")
|
||||
// }
|
||||
}
|
||||
|
||||
private func performLogin() async throws -> Bool {
|
||||
return false
|
||||
// let request = EmailLoginRequest(
|
||||
// email: email,
|
||||
// verificationCode: verificationCode
|
||||
// )
|
||||
//
|
||||
// let apiService = LiveAPIService()
|
||||
// let response: EmailLoginResponse = try await apiService.request(request)
|
||||
//
|
||||
// if response.code == 200, let data = response.data {
|
||||
// // 保存用户信息
|
||||
// await UserInfoManager.saveUserInfo(data)
|
||||
//
|
||||
// // 创建并保存账户模型
|
||||
// let accountModel = AccountModel(
|
||||
// uid: data.uid,
|
||||
// accessToken: data.accessToken,
|
||||
// tokenType: data.tokenType,
|
||||
// refreshToken: data.refreshToken,
|
||||
// expiresIn: data.expiresIn
|
||||
// )
|
||||
// await UserInfoManager.saveAccountModel(accountModel)
|
||||
//
|
||||
// // 获取用户详细信息
|
||||
// if let userInfo = await UserInfoManager.fetchUserInfoFromServer(
|
||||
// uid: String(data.uid),
|
||||
// apiService: apiService
|
||||
// ) {
|
||||
// await UserInfoManager.saveUserInfo(userInfo)
|
||||
// }
|
||||
//
|
||||
// return true
|
||||
// } else {
|
||||
// throw APIError.serverError(response.message ?? "Login failed")
|
||||
// }
|
||||
}
|
||||
|
||||
private func handleCodeRequestResult(_ success: Bool) {
|
||||
isCodeLoading = false
|
||||
if success {
|
||||
startCountdown()
|
||||
}
|
||||
}
|
||||
|
||||
private func handleCodeRequestError(_ error: Error) {
|
||||
isCodeLoading = false
|
||||
errorMessage = error.localizedDescription
|
||||
}
|
||||
|
||||
private func handleLoginResult(_ success: Bool) {
|
||||
isLoading = false
|
||||
if success {
|
||||
loginStep = .completed
|
||||
onLoginSuccess?()
|
||||
}
|
||||
}
|
||||
|
||||
private func handleLoginError(_ error: Error) {
|
||||
isLoading = false
|
||||
errorMessage = error.localizedDescription
|
||||
}
|
||||
|
||||
private func startCountdown() {
|
||||
stopCountdown()
|
||||
codeCountdown = 60
|
||||
timerCancellable = Timer.publish(every: 1.0, on: .main, in: .common)
|
||||
.autoconnect()
|
||||
.sink { _ in
|
||||
if self.codeCountdown > 0 {
|
||||
self.codeCountdown -= 1
|
||||
} else {
|
||||
self.stopCountdown()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func stopCountdown() {
|
||||
timerCancellable?.cancel()
|
||||
timerCancellable = nil
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - EMailLogin View
|
||||
|
||||
struct EMailLoginPage: View {
|
||||
@StateObject private var viewModel = EMailLoginViewModel()
|
||||
let onBack: () -> Void
|
||||
let onLoginSuccess: () -> Void
|
||||
|
||||
@FocusState private var focusedField: Field?
|
||||
|
||||
enum Field {
|
||||
case email
|
||||
case verificationCode
|
||||
}
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
LoginBackgroundView()
|
||||
|
||||
VStack(spacing: 0) {
|
||||
LoginHeaderView(onBack: {
|
||||
viewModel.onBackTapped()
|
||||
})
|
||||
|
||||
Spacer().frame(height: 60)
|
||||
|
||||
Text("Email Login")
|
||||
.font(.system(size: 28, weight: .bold))
|
||||
.foregroundColor(.white)
|
||||
.padding(.bottom, 60)
|
||||
|
||||
VStack(spacing: 24) {
|
||||
// 邮箱输入框
|
||||
emailInputField
|
||||
|
||||
// 验证码输入框(带获取按钮)
|
||||
verificationCodeInputField
|
||||
}
|
||||
.padding(.horizontal, 32)
|
||||
|
||||
Spacer()
|
||||
.frame(height: 80)
|
||||
|
||||
// 登录按钮
|
||||
LoginButtonView(
|
||||
isLoading: viewModel.isLoading,
|
||||
isEnabled: viewModel.isLoginButtonEnabled,
|
||||
onTap: {
|
||||
viewModel.onLoginTapped()
|
||||
}
|
||||
)
|
||||
.padding(.horizontal, 32)
|
||||
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.onAppear {
|
||||
viewModel.onBack = onBack
|
||||
viewModel.onLoginSuccess = onLoginSuccess
|
||||
viewModel.resetState()
|
||||
|
||||
#if DEBUG
|
||||
viewModel.email = "exzero@126.com"
|
||||
#endif
|
||||
}
|
||||
.onDisappear {
|
||||
// viewModel.stopCountdown()
|
||||
}
|
||||
.onChange(of: viewModel.email) { _, newEmail in
|
||||
viewModel.onEmailChanged(newEmail)
|
||||
}
|
||||
.onChange(of: viewModel.verificationCode) { _, newCode in
|
||||
viewModel.onVerificationCodeChanged(newCode)
|
||||
}
|
||||
.onChange(of: viewModel.isCodeLoading) { _, isCodeLoading in
|
||||
if !isCodeLoading && viewModel.errorMessage == nil {
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
|
||||
focusedField = .verificationCode
|
||||
}
|
||||
}
|
||||
}
|
||||
.onChange(of: viewModel.loginStep) { _, newStep in
|
||||
debugInfoSync("🔄 EMailLoginView: loginStep 变化为 \(newStep)")
|
||||
if newStep == .completed {
|
||||
debugInfoSync("✅ EMailLoginView: 登录成功,准备关闭自身")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - UI Components
|
||||
|
||||
private var emailInputField: some View {
|
||||
CustomInputField(
|
||||
type: .text,
|
||||
placeholder: "Please enter email",
|
||||
text: $viewModel.email
|
||||
)
|
||||
.focused($focusedField, equals: .email)
|
||||
}
|
||||
|
||||
private var verificationCodeInputField: some View {
|
||||
CustomInputField(
|
||||
type: .verificationCode,
|
||||
placeholder: "Please enter verification code",
|
||||
text: $viewModel.verificationCode,
|
||||
onGetCode: {
|
||||
viewModel.onGetVerificationCodeTapped()
|
||||
},
|
||||
isCodeButtonEnabled: viewModel.isCodeButtonEnabled,
|
||||
isCodeLoading: viewModel.isCodeLoading,
|
||||
getCodeButtonText: viewModel.getCodeButtonText
|
||||
)
|
||||
.focused($focusedField, equals: .verificationCode)
|
||||
}
|
||||
}
|
||||
|
||||
#Preview {
|
||||
EMailLoginPage(
|
||||
onBack: {},
|
||||
onLoginSuccess: {}
|
||||
)
|
||||
}
|
||||
125
yana/MVVM/IDLoginPage.swift
Normal file
125
yana/MVVM/IDLoginPage.swift
Normal file
@@ -0,0 +1,125 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - IDLogin View
|
||||
|
||||
struct IDLoginPage: View {
|
||||
@StateObject private var viewModel = IDLoginViewModel()
|
||||
let onBack: () -> Void
|
||||
let onLoginSuccess: () -> Void
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
// 背景
|
||||
LoginBackgroundView()
|
||||
|
||||
VStack(spacing: 0) {
|
||||
// 顶部导航栏
|
||||
LoginHeaderView(onBack: {
|
||||
viewModel.onBackTapped()
|
||||
})
|
||||
|
||||
Spacer()
|
||||
.frame(height: 60)
|
||||
|
||||
// 标题
|
||||
Text("ID Login")
|
||||
.font(.system(size: 28, weight: .bold))
|
||||
.foregroundColor(.white)
|
||||
.padding(.bottom, 60)
|
||||
|
||||
// 输入框区域
|
||||
VStack(spacing: 24) {
|
||||
// 用户ID输入框(只允许数字)
|
||||
CustomInputField(
|
||||
type: .number,
|
||||
placeholder: "Please enter ID",
|
||||
text: $viewModel.userID
|
||||
)
|
||||
|
||||
// 密码输入框(带眼睛按钮)
|
||||
CustomInputField(
|
||||
type: .password,
|
||||
placeholder: "Please enter password",
|
||||
text: $viewModel.password,
|
||||
isPasswordVisible: $viewModel.isPasswordVisible
|
||||
)
|
||||
}
|
||||
.padding(.horizontal, 32)
|
||||
|
||||
Spacer()
|
||||
.frame(height: 80)
|
||||
|
||||
// 忘记密码按钮
|
||||
HStack {
|
||||
Spacer()
|
||||
Button(action: {
|
||||
viewModel.onRecoverPasswordTapped()
|
||||
}) {
|
||||
Text("Forgot Password?")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, 32)
|
||||
.padding(.bottom, 20)
|
||||
|
||||
// 登录按钮
|
||||
LoginButtonView(
|
||||
isLoading: viewModel.isLoading || viewModel.isTicketLoading,
|
||||
isEnabled: viewModel.isLoginButtonEnabled,
|
||||
onTap: {
|
||||
viewModel.onLoginTapped()
|
||||
}
|
||||
)
|
||||
.padding(.horizontal, 32)
|
||||
|
||||
// Ticket加载状态提示
|
||||
if viewModel.isTicketLoading {
|
||||
Text("正在获取会话票据...")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
.padding(.top, 8)
|
||||
}
|
||||
|
||||
// 错误信息显示
|
||||
if let errorMessage = viewModel.errorMessage {
|
||||
Text(errorMessage)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
.padding(.top, 8)
|
||||
.padding(.horizontal, 32)
|
||||
}
|
||||
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.navigationDestination(isPresented: $viewModel.showRecoverPassword) {
|
||||
RecoverPasswordPage(
|
||||
onBack: {
|
||||
viewModel.onRecoverPasswordBack()
|
||||
}
|
||||
)
|
||||
.navigationBarHidden(true)
|
||||
}
|
||||
.onAppear {
|
||||
viewModel.onBack = onBack
|
||||
viewModel.onLoginSuccess = onLoginSuccess
|
||||
}
|
||||
.onChange(of: viewModel.loginStep) { _, newStep in
|
||||
debugInfoSync("🔄 IDLoginView: loginStep 变化为 \(newStep)")
|
||||
if newStep == .completed {
|
||||
debugInfoSync("✅ IDLoginView: 登录成功,准备关闭自身")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//#Preview {
|
||||
// IDLoginPage(
|
||||
// onBack: {},
|
||||
// onLoginSuccess: {}
|
||||
// )
|
||||
//}
|
||||
219
yana/MVVM/LoginPage.swift
Normal file
219
yana/MVVM/LoginPage.swift
Normal file
@@ -0,0 +1,219 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - Login ViewModel
|
||||
|
||||
@MainActor
|
||||
class LoginViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var showIDLogin: Bool = false
|
||||
@Published var showEmailLogin: Bool = false
|
||||
@Published var showLanguageSettings: Bool = false
|
||||
@Published var showUserAgreement: Bool = false
|
||||
@Published var showPrivacyPolicy: Bool = false
|
||||
@Published var isAgreementAccepted: Bool = true // 默认选中
|
||||
@Published var showAgreementAlert: Bool = false
|
||||
@Published var isAnyLoginCompleted: Bool = false
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onLoginSuccess: (() -> Void)?
|
||||
private var hasSentSuccess: Bool = false
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onIDLoginTapped() {
|
||||
if isAgreementAccepted {
|
||||
showIDLogin = true
|
||||
} else {
|
||||
showAgreementAlert = true
|
||||
}
|
||||
}
|
||||
|
||||
func onEmailLoginTapped() {
|
||||
if isAgreementAccepted {
|
||||
showEmailLogin = true
|
||||
} else {
|
||||
showAgreementAlert = true
|
||||
}
|
||||
}
|
||||
|
||||
func onLanguageSettingsTapped() {
|
||||
showLanguageSettings = true
|
||||
}
|
||||
|
||||
func onUserAgreementTapped() {
|
||||
showUserAgreement = true
|
||||
}
|
||||
|
||||
func onPrivacyPolicyTapped() {
|
||||
showPrivacyPolicy = true
|
||||
}
|
||||
|
||||
func onLoginCompleted() {
|
||||
guard !hasSentSuccess else { return }
|
||||
isAnyLoginCompleted = true
|
||||
showIDLogin = false
|
||||
showEmailLogin = false
|
||||
hasSentSuccess = true
|
||||
onLoginSuccess?()
|
||||
}
|
||||
|
||||
func onBackFromIDLogin() {
|
||||
showIDLogin = false
|
||||
}
|
||||
|
||||
func onBackFromEmailLogin() {
|
||||
showEmailLogin = false
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Login View
|
||||
|
||||
struct LoginPage: View {
|
||||
@StateObject private var viewModel = LoginViewModel()
|
||||
let onLoginSuccess: () -> Void
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
backgroundView
|
||||
|
||||
VStack(spacing: 0) {
|
||||
Image("top")
|
||||
.resizable()
|
||||
.aspectRatio(375/400, contentMode: .fit)
|
||||
.frame(maxWidth: .infinity)
|
||||
|
||||
HStack {
|
||||
Text(LocalizedString("login.app_title", comment: ""))
|
||||
.font(FontManager.adaptedFont(.bayonRegular, designSize: 56, for: geometry.size.width))
|
||||
.foregroundColor(.white)
|
||||
.padding(.leading, 20)
|
||||
Spacer()
|
||||
}
|
||||
.padding(.bottom, 20)
|
||||
|
||||
Spacer()
|
||||
|
||||
bottomSection
|
||||
}
|
||||
|
||||
// 语言设置按钮 - 固定在页面右上角
|
||||
languageSettingsButton
|
||||
.position(x: geometry.size.width - 40, y: 60)
|
||||
|
||||
APILoadingEffectView()
|
||||
}
|
||||
}
|
||||
.ignoresSafeArea()
|
||||
.navigationBarHidden(true)
|
||||
.navigationDestination(isPresented: $viewModel.showIDLogin) {
|
||||
IDLoginPage(
|
||||
onBack: {
|
||||
viewModel.onBackFromIDLogin()
|
||||
},
|
||||
onLoginSuccess: {
|
||||
viewModel.onLoginCompleted()
|
||||
}
|
||||
)
|
||||
.navigationBarHidden(true)
|
||||
}
|
||||
.navigationDestination(isPresented: $viewModel.showEmailLogin) {
|
||||
EMailLoginPage(
|
||||
onBack: {
|
||||
viewModel.onBackFromEmailLogin()
|
||||
},
|
||||
onLoginSuccess: {
|
||||
viewModel.onLoginCompleted()
|
||||
}
|
||||
)
|
||||
.navigationBarHidden(true)
|
||||
}
|
||||
.sheet(isPresented: $viewModel.showLanguageSettings) {
|
||||
LanguageSettingsView(isPresented: $viewModel.showLanguageSettings)
|
||||
}
|
||||
.webView(
|
||||
isPresented: $viewModel.showUserAgreement,
|
||||
url: APIConfiguration.webURL(for: .userAgreement)
|
||||
)
|
||||
.webView(
|
||||
isPresented: $viewModel.showPrivacyPolicy,
|
||||
url: APIConfiguration.webURL(for: .privacyPolicy)
|
||||
)
|
||||
.alert(LocalizedString("login.agreement_alert_title", comment: ""), isPresented: $viewModel.showAgreementAlert) {
|
||||
Button(LocalizedString("login.agreement_alert_confirm", comment: "")) { }
|
||||
} message: {
|
||||
Text(LocalizedString("login.agreement_alert_message", comment: ""))
|
||||
}
|
||||
.onAppear {
|
||||
viewModel.onLoginSuccess = onLoginSuccess
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 子视图
|
||||
|
||||
private var backgroundView: some View {
|
||||
LoginBackgroundView()
|
||||
}
|
||||
|
||||
private var bottomSection: some View {
|
||||
VStack(spacing: 20) {
|
||||
loginButtons
|
||||
userAgreementComponent
|
||||
}
|
||||
.padding(.horizontal, 28)
|
||||
.padding(.bottom, 48)
|
||||
}
|
||||
|
||||
private var loginButtons: some View {
|
||||
VStack(spacing: 20) {
|
||||
LoginButton(
|
||||
iconName: "person.circle",
|
||||
iconColor: .blue,
|
||||
title: LocalizedString("login.id_login", comment: ""),
|
||||
action: {
|
||||
viewModel.onIDLoginTapped()
|
||||
}
|
||||
)
|
||||
|
||||
LoginButton(
|
||||
iconName: "envelope",
|
||||
iconColor: .green,
|
||||
title: LocalizedString("login.email_login", comment: ""),
|
||||
action: {
|
||||
viewModel.onEmailLoginTapped()
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private var languageSettingsButton: some View {
|
||||
Button(action: {
|
||||
viewModel.onLanguageSettingsTapped()
|
||||
}) {
|
||||
Image(systemName: "globe")
|
||||
.font(.system(size: 20))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
}
|
||||
|
||||
private var userAgreementComponent: some View {
|
||||
UserAgreementComponent(
|
||||
isAgreed: $viewModel.isAgreementAccepted,
|
||||
onAgreementTap: {
|
||||
Task { @MainActor in
|
||||
viewModel.onUserAgreementTapped()
|
||||
}
|
||||
},
|
||||
onPolicyTap: {
|
||||
Task { @MainActor in
|
||||
viewModel.onPrivacyPolicyTapped()
|
||||
}
|
||||
}
|
||||
)
|
||||
.frame(height: 40)
|
||||
.padding(.horizontal, -20)
|
||||
}
|
||||
}
|
||||
|
||||
#Preview {
|
||||
LoginPage(onLoginSuccess: {})
|
||||
}
|
||||
58
yana/MVVM/MainPage.swift
Normal file
58
yana/MVVM/MainPage.swift
Normal file
@@ -0,0 +1,58 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - Main View
|
||||
|
||||
struct MainPage: View {
|
||||
@StateObject private var viewModel = MainViewModel()
|
||||
let onLogout: () -> Void
|
||||
@State private var isPresentingCreatePage: Bool = false
|
||||
|
||||
var body: some View {
|
||||
NavigationStack(path: $viewModel.navigationPath) {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
// 背景图片
|
||||
LoginBackgroundView()
|
||||
|
||||
// 主内容:使用 TabView 常驻子树
|
||||
TabView(selection: $viewModel.selectedTab) {
|
||||
MomentListHomePage(onCreateTapped: { isPresentingCreatePage = true })
|
||||
.tag(MainViewModel.Tab.feed)
|
||||
MePage(onLogout: onLogout)
|
||||
.tag(MainViewModel.Tab.me)
|
||||
}
|
||||
.tabViewStyle(.page(indexDisplayMode: .never))
|
||||
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||
|
||||
VStack {
|
||||
Spacer()
|
||||
// 底部导航栏(组件化)
|
||||
BottomTabBar(viewModel: viewModel)
|
||||
.frame(height: 80)
|
||||
.padding(.horizontal, 24)
|
||||
.padding(.bottom)
|
||||
}
|
||||
}.ignoresSafeArea(.all)
|
||||
}
|
||||
.toolbar(.hidden)
|
||||
}
|
||||
.onAppear {
|
||||
viewModel.onLogout = onLogout
|
||||
viewModel.onAddButtonTapped = {
|
||||
// TODO: 处理添加按钮点击事件
|
||||
debugInfoSync("➕ 添加按钮被点击")
|
||||
}
|
||||
viewModel.onAppear()
|
||||
}
|
||||
.fullScreenCover(isPresented: $isPresentingCreatePage) {
|
||||
CreateFeedPage {
|
||||
isPresentingCreatePage = false
|
||||
}
|
||||
}
|
||||
.onChange(of: viewModel.isLoggedOut) { _, isLoggedOut in
|
||||
if isLoggedOut {
|
||||
onLogout()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
186
yana/MVVM/MePage.swift
Normal file
186
yana/MVVM/MePage.swift
Normal file
@@ -0,0 +1,186 @@
|
||||
import SwiftUI
|
||||
|
||||
struct MePage: View {
|
||||
let onLogout: () -> Void
|
||||
@State private var isShowingSettings: Bool = false
|
||||
@StateObject private var viewModel = MePageViewModel()
|
||||
|
||||
// 图片预览状态
|
||||
@State private var previewItem: PreviewItem? = nil
|
||||
@State private var previewCurrentIndex: Int = 0
|
||||
|
||||
// 详情页状态
|
||||
@State private var selectedMoment: MomentsInfo? = nil
|
||||
|
||||
var body: some View {
|
||||
ZStack {
|
||||
// 背景
|
||||
// MomentListBackgroundView()
|
||||
|
||||
VStack(spacing: 0) {
|
||||
// 顶部:大头像 + 姓名 + ID + 右上角设置
|
||||
ZStack(alignment: .topTrailing) {
|
||||
VStack(spacing: 12) {
|
||||
AsyncImage(url: URL(string: viewModel.avatarURL)) { image in
|
||||
image.resizable().scaledToFill()
|
||||
} placeholder: {
|
||||
Image(systemName: "person.circle.fill")
|
||||
.resizable()
|
||||
.scaledToFill()
|
||||
.foregroundColor(.gray)
|
||||
}
|
||||
.frame(width: 132, height: 132)
|
||||
.clipShape(Circle())
|
||||
.overlay(Circle().stroke(Color.white, lineWidth: 3))
|
||||
.shadow(color: .black.opacity(0.25), radius: 10, x: 0, y: 6)
|
||||
|
||||
Text(viewModel.nickname.isEmpty ? "未知用户" : viewModel.nickname)
|
||||
.font(.system(size: 34, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
.lineLimit(1)
|
||||
.minimumScaleFactor(0.6)
|
||||
|
||||
if viewModel.userId > 0 {
|
||||
HStack(spacing: 6) {
|
||||
Text("ID:\(viewModel.userId)")
|
||||
.font(.system(size: 16))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
Image(systemName: "doc.on.doc")
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
}
|
||||
}
|
||||
.frame(maxWidth: .infinity)
|
||||
.padding(.top, 24)
|
||||
|
||||
Button(action: { isShowingSettings = true }) {
|
||||
Image(systemName: "gearshape")
|
||||
.font(.system(size: 24, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.frame(width: 40, height: 40)
|
||||
.background(Color.black.opacity(0.3))
|
||||
.clipShape(Circle())
|
||||
}
|
||||
.padding(.trailing, 16)
|
||||
.padding(.top, 8)
|
||||
}
|
||||
.padding(.bottom, 8)
|
||||
|
||||
// 下部:只显示当前用户的动态列表
|
||||
if !viewModel.moments.isEmpty {
|
||||
ScrollView {
|
||||
LazyVStack(spacing: 16) {
|
||||
ForEach(Array(viewModel.moments.enumerated()), id: \.offset) { index, moment in
|
||||
MomentListItem(
|
||||
moment: moment,
|
||||
onImageTap: { images, tappedIndex in
|
||||
previewCurrentIndex = tappedIndex
|
||||
previewItem = PreviewItem(images: images, index: tappedIndex)
|
||||
},
|
||||
onMomentTap: { tapped in
|
||||
selectedMoment = tapped
|
||||
debugInfoSync("➡️ MePage: 动态被点击 - ID: \(tapped.dynamicId)")
|
||||
}
|
||||
)
|
||||
.padding(.horizontal, 16)
|
||||
.onAppear {
|
||||
if index == viewModel.moments.count - 3 {
|
||||
viewModel.loadMoreData()
|
||||
}
|
||||
}
|
||||
}
|
||||
if viewModel.isLoadingMore {
|
||||
HStack {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
Text("加载更多...")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
.padding(.vertical, 20)
|
||||
}
|
||||
if !viewModel.hasMore && !viewModel.moments.isEmpty {
|
||||
Text("没有更多数据了")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
.padding(.vertical, 20)
|
||||
}
|
||||
}
|
||||
.padding(.bottom, 160)
|
||||
}
|
||||
.refreshable { await viewModel.refreshData() }
|
||||
} else if viewModel.isLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.padding(.top, 20)
|
||||
} else if let error = viewModel.errorMessage {
|
||||
VStack(spacing: 16) {
|
||||
Text(error)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
.multilineTextAlignment(.center)
|
||||
.padding(.horizontal, 20)
|
||||
Button(action: { Task { await viewModel.refreshData() } }) {
|
||||
Text("重试")
|
||||
.font(.system(size: 14, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.padding(.horizontal, 20)
|
||||
.padding(.vertical, 8)
|
||||
.background(Color.white.opacity(0.2))
|
||||
.cornerRadius(8)
|
||||
}
|
||||
}
|
||||
.padding(.top, 20)
|
||||
} else {
|
||||
VStack(spacing: 12) {
|
||||
Image(systemName: "doc.text")
|
||||
.font(.system(size: 32))
|
||||
.foregroundColor(.white.opacity(0.5))
|
||||
Text("暂无动态")
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white.opacity(0.7))
|
||||
}
|
||||
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||
}
|
||||
Spacer()
|
||||
}
|
||||
.safeAreaPadding(.top, 8)
|
||||
}
|
||||
.onAppear { viewModel.onAppear() }
|
||||
.onReceive(NotificationCenter.default.publisher(for: .init("CreateFeedPublished"))) { _ in
|
||||
Task { await viewModel.refreshData() }
|
||||
}
|
||||
.sheet(isPresented: $isShowingSettings) {
|
||||
SettingPage(
|
||||
onBack: { isShowingSettings = false },
|
||||
onLogout: {
|
||||
isShowingSettings = false
|
||||
onLogout()
|
||||
}
|
||||
)
|
||||
.navigationBarHidden(true)
|
||||
}
|
||||
// 图片预览
|
||||
.sheet(item: $previewItem) { item in
|
||||
ImagePreviewPager(
|
||||
images: item.images as [String],
|
||||
currentIndex: $previewCurrentIndex
|
||||
) {
|
||||
previewItem = nil
|
||||
}
|
||||
}
|
||||
// 详情页
|
||||
.sheet(item: $selectedMoment) { moment in
|
||||
MomentDetailPage(moment: moment) {
|
||||
selectedMoment = nil
|
||||
debugInfoSync("📱 MePage: 详情页已关闭")
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.presentationDetents([.large])
|
||||
.presentationDragIndicator(.visible)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
245
yana/MVVM/MomentDetailPage.swift
Normal file
245
yana/MVVM/MomentDetailPage.swift
Normal file
@@ -0,0 +1,245 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - MomentDetailPage
|
||||
|
||||
struct MomentDetailPage: View {
|
||||
@StateObject private var viewModel: MomentDetailViewModel
|
||||
let onClose: () -> Void
|
||||
|
||||
init(moment: MomentsInfo, onClose: @escaping () -> Void) {
|
||||
_viewModel = StateObject(wrappedValue: MomentDetailViewModel(moment: moment))
|
||||
self.onClose = onClose
|
||||
}
|
||||
|
||||
var body: some View {
|
||||
ZStack {
|
||||
// 背景
|
||||
LoginBackgroundView()
|
||||
.ignoresSafeArea()
|
||||
|
||||
VStack(spacing: 0) {
|
||||
// 顶部导航栏
|
||||
HStack {
|
||||
Button {
|
||||
onClose()
|
||||
} label: {
|
||||
Image(systemName: "chevron.left")
|
||||
.font(.system(size: 20, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.frame(width: 44, height: 44)
|
||||
.background(Color.black.opacity(0.3))
|
||||
.clipShape(Circle())
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
Text(LocalizedString("detail.title", comment: "Detail page title"))
|
||||
.font(.system(size: 18, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
.shadow(color: .black.opacity(0.5), radius: 2, x: 0, y: 1)
|
||||
|
||||
Spacer()
|
||||
|
||||
// 占位,保持标题居中
|
||||
Color.clear
|
||||
.frame(width: 44, height: 44)
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.safeAreaPadding(.top, 60)
|
||||
.padding(.bottom, 12)
|
||||
.background(
|
||||
LinearGradient(
|
||||
gradient: Gradient(colors: [
|
||||
Color.black.opacity(0.4),
|
||||
Color.black.opacity(0.2),
|
||||
Color.clear
|
||||
]),
|
||||
startPoint: .top,
|
||||
endPoint: .bottom
|
||||
)
|
||||
)
|
||||
|
||||
// 内容区域
|
||||
ScrollView {
|
||||
VStack(alignment: .leading, spacing: 12) {
|
||||
// 用户信息
|
||||
HStack(alignment: .top) {
|
||||
// 头像
|
||||
CachedAsyncImage(url: viewModel.moment.avatar) { image in
|
||||
image
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
} placeholder: {
|
||||
Circle()
|
||||
.fill(Color.gray.opacity(0.3))
|
||||
.overlay(
|
||||
Text(String(viewModel.moment.nick.prefix(1)))
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
)
|
||||
}
|
||||
.frame(width: 44, height: 44)
|
||||
.clipShape(Circle())
|
||||
|
||||
VStack(alignment: .leading, spacing: 2) {
|
||||
Text(viewModel.moment.nick)
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
UserIDDisplay(uid: viewModel.moment.uid, fontSize: 12, textColor: .white.opacity(0.6))
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
// 时间
|
||||
Text(formatDisplayTime(viewModel.moment.publishTime))
|
||||
.font(.system(size: 12, weight: .bold))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
.padding(.horizontal, 6)
|
||||
.padding(.vertical, 2)
|
||||
.background(Color.white.opacity(0.15))
|
||||
.cornerRadius(4)
|
||||
}
|
||||
|
||||
// 动态内容
|
||||
if !viewModel.moment.content.isEmpty {
|
||||
Text(viewModel.moment.content)
|
||||
.font(.system(size: 16))
|
||||
.foregroundColor(.white.opacity(0.95))
|
||||
.multilineTextAlignment(.leading)
|
||||
}
|
||||
|
||||
// 图片网格
|
||||
if let images = viewModel.moment.dynamicResList, !images.isEmpty {
|
||||
MomentImageGrid(
|
||||
images: images,
|
||||
onImageTap: { images, index in
|
||||
viewModel.onImageTap(index)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
// 互动按钮
|
||||
HStack(spacing: 20) {
|
||||
Button {
|
||||
viewModel.like()
|
||||
} label: {
|
||||
HStack(spacing: 6) {
|
||||
if viewModel.isLikeLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: viewModel.localIsLike ? .red : .white.opacity(0.8)))
|
||||
.scaleEffect(0.8)
|
||||
} else {
|
||||
Image(systemName: viewModel.localIsLike ? "heart.fill" : "heart")
|
||||
.font(.system(size: 18))
|
||||
}
|
||||
Text("\(viewModel.localLikeCount)")
|
||||
.font(.system(size: 16))
|
||||
}
|
||||
}
|
||||
.foregroundColor(viewModel.localIsLike ? .red : .white.opacity(0.9))
|
||||
.disabled(viewModel.isLikeLoading || viewModel.moment.status == 0)
|
||||
.opacity(viewModel.moment.status == 0 ? 0.5 : 1.0)
|
||||
|
||||
Spacer()
|
||||
|
||||
// 审核中状态角标 - 与外部列表保持一致:右侧对齐并与点赞按钮居中对齐
|
||||
if viewModel.moment.status == 0 {
|
||||
Text("reviewing")
|
||||
.font(.system(size: 12, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
.padding(.horizontal, 10)
|
||||
.padding(.vertical, 6)
|
||||
.background(Color.orange.opacity(0.85))
|
||||
.clipShape(Capsule())
|
||||
}
|
||||
}
|
||||
.padding(.top, 8)
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.bottom, 16)
|
||||
.safeAreaPadding(.top, 8)
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.fullScreenCover(isPresented: $viewModel.showImagePreview) {
|
||||
ImagePreviewPager(
|
||||
images: viewModel.images,
|
||||
currentIndex: $viewModel.currentIndex
|
||||
) {
|
||||
viewModel.showImagePreview = false
|
||||
}
|
||||
}
|
||||
.onAppear {
|
||||
debugInfoSync("📱 MomentDetailPage: 显示详情页")
|
||||
debugInfoSync(" 动态ID: \(viewModel.moment.dynamicId)")
|
||||
debugInfoSync(" 用户: \(viewModel.moment.nick)")
|
||||
debugInfoSync(" 审核状态: \(viewModel.moment.status)")
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 时间显示逻辑
|
||||
private func formatDisplayTime(_ timestamp: Int) -> String {
|
||||
let date = Date(timeIntervalSince1970: TimeInterval(timestamp) / 1000.0)
|
||||
let formatter = DateFormatter()
|
||||
formatter.locale = Locale(identifier: "zh_CN")
|
||||
let now = Date()
|
||||
let interval = now.timeIntervalSince(date)
|
||||
let calendar = Calendar.current
|
||||
if calendar.isDateInToday(date) {
|
||||
if interval < 60 {
|
||||
return "刚刚"
|
||||
} else if interval < 3600 {
|
||||
return "\(Int(interval / 60))分钟前"
|
||||
} else {
|
||||
return "\(Int(interval / 3600))小时前"
|
||||
}
|
||||
} else {
|
||||
formatter.dateFormat = "MM/dd"
|
||||
return formatter.string(from: date)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//#Preview {
|
||||
// let testMoment = MomentsInfo(
|
||||
// dynamicId: 1,
|
||||
// uid: 123456,
|
||||
// nick: "测试用户",
|
||||
// avatar: "",
|
||||
// type: 0,
|
||||
// content: "这是一条测试动态内容,用来测试 MomentDetailPage 的显示效果。",
|
||||
// likeCount: 42,
|
||||
// isLike: false,
|
||||
// commentCount: 5,
|
||||
// publishTime: Int(Date().timeIntervalSince1970 * 1000),
|
||||
// worldId: 1,
|
||||
// status: 0, // 审核中状态
|
||||
// playCount: nil,
|
||||
// dynamicResList: [
|
||||
// MomentsPicture(id: 1, resUrl: "https://picsum.photos/300/300", format: "jpg", width: 300, height: 300, resDuration: nil),
|
||||
// MomentsPicture(id: 2, resUrl: "https://picsum.photos/301/301", format: "jpg", width: 301, height: 301, resDuration: nil)
|
||||
// ],
|
||||
// gender: nil,
|
||||
// squareTop: nil,
|
||||
// topicTop: nil,
|
||||
// newUser: nil,
|
||||
// defUser: nil,
|
||||
// scene: nil,
|
||||
// userVipInfoVO: nil,
|
||||
// headwearPic: nil,
|
||||
// headwearEffect: nil,
|
||||
// headwearType: nil,
|
||||
// headwearName: nil,
|
||||
// headwearId: nil,
|
||||
// experLevelPic: nil,
|
||||
// charmLevelPic: nil,
|
||||
// isCustomWord: nil,
|
||||
// labelList: nil
|
||||
// )
|
||||
//
|
||||
// MomentDetailPage(moment: testMoment) {
|
||||
// print("关闭详情页")
|
||||
// }
|
||||
//}
|
||||
436
yana/MVVM/RecoverPasswordPage.swift
Normal file
436
yana/MVVM/RecoverPasswordPage.swift
Normal file
@@ -0,0 +1,436 @@
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
// MARK: - RecoverPassword ViewModel
|
||||
|
||||
@MainActor
|
||||
class RecoverPasswordViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var email: String = ""
|
||||
@Published var verificationCode: String = ""
|
||||
@Published var newPassword: String = ""
|
||||
@Published var isNewPasswordVisible: Bool = false
|
||||
@Published var countdown: Int = 0
|
||||
@Published var isResetLoading: Bool = false
|
||||
@Published var isCodeLoading: Bool = false
|
||||
@Published var errorMessage: String?
|
||||
@Published var isResetSuccess: Bool = false
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onBack: (() -> Void)?
|
||||
|
||||
// MARK: - Private Properties
|
||||
private var timerCancellable: AnyCancellable?
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var isEmailValid: Bool {
|
||||
!email.isEmpty
|
||||
}
|
||||
|
||||
var isVerificationCodeValid: Bool {
|
||||
!verificationCode.isEmpty
|
||||
}
|
||||
|
||||
var isNewPasswordValid: Bool {
|
||||
!newPassword.isEmpty
|
||||
}
|
||||
|
||||
var isConfirmButtonEnabled: Bool {
|
||||
!isResetLoading && isEmailValid && isVerificationCodeValid && isNewPasswordValid
|
||||
}
|
||||
|
||||
var isGetCodeButtonEnabled: Bool {
|
||||
!isCodeLoading && isEmailValid && countdown == 0
|
||||
}
|
||||
|
||||
var getCodeButtonText: String {
|
||||
if isCodeLoading {
|
||||
return ""
|
||||
} else if countdown > 0 {
|
||||
return "\(countdown)s"
|
||||
} else {
|
||||
return LocalizedString("recover_password.get_code", comment: "")
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onBackTapped() {
|
||||
onBack?()
|
||||
}
|
||||
|
||||
func onEmailChanged(_ newEmail: String) {
|
||||
email = newEmail
|
||||
}
|
||||
|
||||
func onVerificationCodeChanged(_ newCode: String) {
|
||||
verificationCode = newCode
|
||||
}
|
||||
|
||||
func onNewPasswordChanged(_ newPassword: String) {
|
||||
self.newPassword = newPassword
|
||||
}
|
||||
|
||||
func onGetVerificationCodeTapped() {
|
||||
guard isGetCodeButtonEnabled else { return }
|
||||
|
||||
isCodeLoading = true
|
||||
errorMessage = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let result = try await requestVerificationCode()
|
||||
await MainActor.run {
|
||||
self.handleCodeRequestResult(result)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleCodeRequestError(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func onResetPasswordTapped() {
|
||||
guard isConfirmButtonEnabled else { return }
|
||||
|
||||
isResetLoading = true
|
||||
errorMessage = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let result = try await resetPassword()
|
||||
await MainActor.run {
|
||||
self.handleResetResult(result)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleResetError(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func resetState() {
|
||||
email = ""
|
||||
verificationCode = ""
|
||||
newPassword = ""
|
||||
isNewPasswordVisible = false
|
||||
countdown = 0
|
||||
isResetLoading = false
|
||||
isCodeLoading = false
|
||||
errorMessage = nil
|
||||
isResetSuccess = false
|
||||
stopCountdown()
|
||||
|
||||
#if DEBUG
|
||||
email = "exzero@126.com"
|
||||
#endif
|
||||
}
|
||||
|
||||
// MARK: - Private Methods
|
||||
private func requestVerificationCode() async throws -> Bool {
|
||||
return false
|
||||
// let request = EmailVerificationCodeRequest(email: email)
|
||||
// let apiService = LiveAPIService()
|
||||
// let response: EmailVerificationCodeResponse = try await apiService.request(request)
|
||||
//
|
||||
// if response.code == 200 {
|
||||
// return true
|
||||
// } else {
|
||||
// throw APIError.serverError(response.message ?? "Failed to send verification code")
|
||||
// }
|
||||
}
|
||||
|
||||
private func resetPassword() async throws -> Bool {
|
||||
return false
|
||||
// let request = ResetPasswordRequest(
|
||||
// email: email,
|
||||
// verificationCode: verificationCode,
|
||||
// newPassword: newPassword
|
||||
// )
|
||||
//
|
||||
// let apiService = LiveAPIService()
|
||||
// let response: ResetPasswordResponse = try await apiService.request(request)
|
||||
//
|
||||
// if response.code == 200 {
|
||||
// return true
|
||||
// } else {
|
||||
// throw APIError.serverError(response.message ?? "Failed to reset password")
|
||||
// }
|
||||
}
|
||||
|
||||
private func handleCodeRequestResult(_ success: Bool) {
|
||||
isCodeLoading = false
|
||||
if success {
|
||||
startCountdown()
|
||||
}
|
||||
}
|
||||
|
||||
private func handleCodeRequestError(_ error: Error) {
|
||||
isCodeLoading = false
|
||||
errorMessage = error.localizedDescription
|
||||
}
|
||||
|
||||
private func handleResetResult(_ success: Bool) {
|
||||
isResetLoading = false
|
||||
if success {
|
||||
isResetSuccess = true
|
||||
onBack?()
|
||||
}
|
||||
}
|
||||
|
||||
private func handleResetError(_ error: Error) {
|
||||
isResetLoading = false
|
||||
errorMessage = error.localizedDescription
|
||||
}
|
||||
|
||||
private func startCountdown() {
|
||||
stopCountdown()
|
||||
countdown = 60
|
||||
|
||||
timerCancellable = Timer.publish(every: 1.0, on: .main, in: .common)
|
||||
.autoconnect()
|
||||
.sink { _ in
|
||||
if self.countdown > 0 {
|
||||
self.countdown -= 1
|
||||
} else {
|
||||
self.stopCountdown()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func stopCountdown() {
|
||||
timerCancellable?.cancel()
|
||||
timerCancellable = nil
|
||||
countdown = 0
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - RecoverPassword View
|
||||
|
||||
struct RecoverPasswordPage: View {
|
||||
@StateObject private var viewModel = RecoverPasswordViewModel()
|
||||
let onBack: () -> Void
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
// 背景图片
|
||||
LoginBackgroundView()
|
||||
|
||||
VStack(spacing: 0) {
|
||||
// 顶部导航栏
|
||||
LoginHeaderView(onBack: {
|
||||
viewModel.onBackTapped()
|
||||
})
|
||||
|
||||
Spacer()
|
||||
.frame(height: 60)
|
||||
|
||||
// 标题
|
||||
Text(LocalizedString("recover_password.title", comment: ""))
|
||||
.font(.system(size: 28, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.padding(.bottom, 80)
|
||||
|
||||
// 输入框区域
|
||||
VStack(spacing: 24) {
|
||||
// 邮箱输入框
|
||||
emailInputField
|
||||
|
||||
// 验证码输入框(带获取按钮)
|
||||
verificationCodeInputField
|
||||
|
||||
// 新密码输入框
|
||||
newPasswordInputField
|
||||
}
|
||||
.padding(.horizontal, 32)
|
||||
|
||||
Spacer()
|
||||
.frame(height: 80)
|
||||
|
||||
// 确认按钮
|
||||
confirmButton
|
||||
|
||||
// 错误信息
|
||||
if let errorMessage = viewModel.errorMessage {
|
||||
Text(errorMessage)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
.padding(.top, 16)
|
||||
.padding(.horizontal, 32)
|
||||
}
|
||||
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
}
|
||||
.onAppear {
|
||||
viewModel.onBack = onBack
|
||||
viewModel.resetState()
|
||||
}
|
||||
.onDisappear {
|
||||
// viewModel.stopCountdown()
|
||||
}
|
||||
.onChange(of: viewModel.email) { _, newEmail in
|
||||
viewModel.onEmailChanged(newEmail)
|
||||
}
|
||||
.onChange(of: viewModel.verificationCode) { _, newCode in
|
||||
viewModel.onVerificationCodeChanged(newCode)
|
||||
}
|
||||
.onChange(of: viewModel.newPassword) { _, newPassword in
|
||||
viewModel.onNewPasswordChanged(newPassword)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - UI Components
|
||||
|
||||
private var emailInputField: some View {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.fill(Color.white.opacity(0.1))
|
||||
.overlay(
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.stroke(Color.white.opacity(0.3), lineWidth: 1)
|
||||
)
|
||||
.frame(height: 56)
|
||||
|
||||
TextField("", text: $viewModel.email)
|
||||
.placeholder(when: viewModel.email.isEmpty) {
|
||||
Text(LocalizedString("recover_password.placeholder_email", comment: ""))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 16))
|
||||
.padding(.horizontal, 24)
|
||||
.keyboardType(.emailAddress)
|
||||
.autocapitalization(.none)
|
||||
}
|
||||
}
|
||||
|
||||
private var verificationCodeInputField: some View {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.fill(Color.white.opacity(0.1))
|
||||
.overlay(
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.stroke(Color.white.opacity(0.3), lineWidth: 1)
|
||||
)
|
||||
.frame(height: 56)
|
||||
|
||||
HStack {
|
||||
TextField("", text: $viewModel.verificationCode)
|
||||
.placeholder(when: viewModel.verificationCode.isEmpty) {
|
||||
Text(LocalizedString("recover_password.placeholder_verification_code", comment: ""))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 16))
|
||||
.keyboardType(.numberPad)
|
||||
|
||||
// 获取验证码按钮
|
||||
Button(action: {
|
||||
viewModel.onGetVerificationCodeTapped()
|
||||
}) {
|
||||
ZStack {
|
||||
if viewModel.isCodeLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.7)
|
||||
} else {
|
||||
Text(viewModel.getCodeButtonText)
|
||||
.font(.system(size: 14, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
}
|
||||
}
|
||||
.frame(width: 60, height: 36)
|
||||
.background(
|
||||
RoundedRectangle(cornerRadius: 15)
|
||||
.fill(Color.white.opacity(viewModel.isGetCodeButtonEnabled ? 0.2 : 0.1))
|
||||
)
|
||||
}
|
||||
.disabled(!viewModel.isGetCodeButtonEnabled || viewModel.isCodeLoading)
|
||||
}
|
||||
.padding(.horizontal, 24)
|
||||
}
|
||||
}
|
||||
|
||||
private var newPasswordInputField: some View {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.fill(Color.white.opacity(0.1))
|
||||
.overlay(
|
||||
RoundedRectangle(cornerRadius: 25)
|
||||
.stroke(Color.white.opacity(0.3), lineWidth: 1)
|
||||
)
|
||||
.frame(height: 56)
|
||||
|
||||
HStack {
|
||||
if viewModel.isNewPasswordVisible {
|
||||
TextField("", text: $viewModel.newPassword)
|
||||
.placeholder(when: viewModel.newPassword.isEmpty) {
|
||||
Text(LocalizedString("recover_password.placeholder_new_password", comment: ""))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 16))
|
||||
} else {
|
||||
SecureField("", text: $viewModel.newPassword)
|
||||
.placeholder(when: viewModel.newPassword.isEmpty) {
|
||||
Text(LocalizedString("recover_password.placeholder_new_password", comment: ""))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
}
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 16))
|
||||
}
|
||||
|
||||
Button(action: {
|
||||
viewModel.isNewPasswordVisible.toggle()
|
||||
}) {
|
||||
Image(systemName: viewModel.isNewPasswordVisible ? "eye.slash" : "eye")
|
||||
.foregroundColor(.white.opacity(0.7))
|
||||
.font(.system(size: 18))
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, 24)
|
||||
}
|
||||
}
|
||||
|
||||
private var confirmButton: some View {
|
||||
Button(action: {
|
||||
viewModel.onResetPasswordTapped()
|
||||
}) {
|
||||
ZStack {
|
||||
// 渐变背景
|
||||
LinearGradient(
|
||||
colors: [
|
||||
Color(red: 0.85, green: 0.37, blue: 1.0), // #D85EFF
|
||||
Color(red: 0.54, green: 0.31, blue: 1.0) // #8A4FFF
|
||||
],
|
||||
startPoint: .leading,
|
||||
endPoint: .trailing
|
||||
)
|
||||
.clipShape(RoundedRectangle(cornerRadius: 28))
|
||||
|
||||
HStack {
|
||||
if viewModel.isResetLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
}
|
||||
Text(viewModel.isResetLoading ? LocalizedString("recover_password.resetting", comment: "") : LocalizedString("recover_password.confirm_button", comment: ""))
|
||||
.font(.system(size: 18, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
}
|
||||
}
|
||||
.frame(height: 56)
|
||||
}
|
||||
.disabled(!viewModel.isConfirmButtonEnabled)
|
||||
.opacity(viewModel.isConfirmButtonEnabled ? 1.0 : 0.5)
|
||||
.padding(.horizontal, 32)
|
||||
}
|
||||
}
|
||||
|
||||
#Preview {
|
||||
RecoverPasswordPage(onBack: {})
|
||||
}
|
||||
62
yana/MVVM/SplashPage.swift
Normal file
62
yana/MVVM/SplashPage.swift
Normal file
@@ -0,0 +1,62 @@
|
||||
import SwiftUI
|
||||
|
||||
struct SplashPage: View {
|
||||
@State private var showLogin = false
|
||||
@State private var showMain = false
|
||||
@State private var hasCheckedAuth = false
|
||||
private let splashTransitionAnimation: Animation = .easeInOut(duration: 0.5)
|
||||
|
||||
var body: some View {
|
||||
Group {
|
||||
if showMain {
|
||||
MainPage(onLogout: {
|
||||
showMain = false
|
||||
showLogin = true
|
||||
})
|
||||
} else if showLogin {
|
||||
NavigationStack {
|
||||
LoginPage(onLoginSuccess: {
|
||||
showMain = true
|
||||
})
|
||||
}
|
||||
} else {
|
||||
ZStack {
|
||||
LoginBackgroundView()
|
||||
VStack(spacing: 32) {
|
||||
Spacer().frame(height: 200)
|
||||
Image("logo")
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fit)
|
||||
.frame(width: 100, height: 100)
|
||||
Text(LocalizedString("splash.title", comment: "E-Parti"))
|
||||
.font(.system(size: 40, weight: .regular))
|
||||
.foregroundColor(.white)
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
.onAppear {
|
||||
guard !hasCheckedAuth else { return }
|
||||
hasCheckedAuth = true
|
||||
Task { @MainActor in
|
||||
debugInfoSync("🚀 SplashV2 启动,开始检查登录缓存")
|
||||
let status = await UserInfoManager.checkAuthenticationStatus()
|
||||
if status.canAutoLogin {
|
||||
debugInfoSync("✅ 检测到可自动登录,尝试预取用户信息")
|
||||
_ = await UserInfoManager.autoFetchUserInfoOnAppLaunch(apiService: LiveAPIService())
|
||||
withAnimation(splashTransitionAnimation) {
|
||||
showMain = true
|
||||
}
|
||||
} else {
|
||||
debugInfoSync("🔑 未登录或缓存无效,进入登录页")
|
||||
withAnimation(splashTransitionAnimation) {
|
||||
showLogin = true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
203
yana/MVVM/View/MomentListHomePage.swift
Normal file
203
yana/MVVM/View/MomentListHomePage.swift
Normal file
@@ -0,0 +1,203 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - BackgroundView
|
||||
struct MomentListBackgroundView: View {
|
||||
var body: some View {
|
||||
Image("bg")
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||
.clipped()
|
||||
.ignoresSafeArea(.all)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - MomentListHomePage
|
||||
struct MomentListHomePage: View {
|
||||
@StateObject private var viewModel = MomentListHomeViewModel()
|
||||
let onCreateTapped: () -> Void
|
||||
|
||||
// MARK: - 图片预览状态
|
||||
@State private var previewItem: PreviewItem? = nil
|
||||
@State private var previewCurrentIndex: Int = 0
|
||||
|
||||
// MARK: - 详情页状态
|
||||
@State private var selectedMoment: MomentsInfo? = nil
|
||||
|
||||
// MARK: - 创建动态发布页弹窗
|
||||
// 迁移到上层(MainPage)统一管理,避免与 TabView 全屏弹窗冲突
|
||||
|
||||
var body: some View {
|
||||
ZStack {
|
||||
// 背景
|
||||
// MomentListBackgroundView()
|
||||
|
||||
VStack(alignment: .center, spacing: 0) {
|
||||
// 顶部标题居中 + 右上角添加按钮(垂直居中对齐)
|
||||
ZStack {
|
||||
// 居中标题
|
||||
Text(LocalizedString("feedList.title", comment: "Enjoy your Life Time"))
|
||||
.font(.system(size: 22, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
.frame(maxWidth: .infinity, alignment: .center)
|
||||
|
||||
// 右上角 “+” 按钮
|
||||
HStack {
|
||||
Spacer()
|
||||
Button {
|
||||
debugInfoSync("➕ MomentListHomePage: 点击添加按钮")
|
||||
onCreateTapped()
|
||||
} label: {
|
||||
Image("add icon")
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fit)
|
||||
.frame(width: 40, height: 40)
|
||||
}
|
||||
.padding(.trailing, 16)
|
||||
}
|
||||
}
|
||||
.frame(height: 56)
|
||||
|
||||
// 动态列表内容(Volume 与标语随列表滚动)
|
||||
if !viewModel.moments.isEmpty {
|
||||
ScrollView {
|
||||
VStack(spacing: 0) {
|
||||
// Volume 图标 + 标语(随列表滚动)
|
||||
Image("Volume")
|
||||
.frame(width: 56, height: 41)
|
||||
.padding(.top, 16)
|
||||
Text(LocalizedString("feedList.slogan",
|
||||
comment: ""))
|
||||
.font(.system(size: 16))
|
||||
.multilineTextAlignment(.leading)
|
||||
.foregroundColor(.white.opacity(0.9))
|
||||
.padding(.horizontal, 30)
|
||||
.padding(.bottom, 30)
|
||||
|
||||
LazyVStack(spacing: 16) {
|
||||
ForEach(Array(viewModel.moments.enumerated()), id: \.element.dynamicId) { index, moment in
|
||||
MomentListItem(
|
||||
moment: moment,
|
||||
onImageTap: { images, tappedIndex in
|
||||
// 处理图片点击事件
|
||||
previewCurrentIndex = tappedIndex
|
||||
previewItem = PreviewItem(images: images, index: tappedIndex)
|
||||
debugInfoSync("📸 MomentListHomePage: 图片被点击")
|
||||
debugInfoSync(" 动态索引: \(index)")
|
||||
debugInfoSync(" 图片索引: \(tappedIndex)")
|
||||
debugInfoSync(" 图片数量: \(images.count)")
|
||||
},
|
||||
onMomentTap: { tappedMoment in
|
||||
// 处理整体点击事件 - 打开详情页
|
||||
selectedMoment = tappedMoment
|
||||
debugInfoSync("➡️ MomentListHomePage: 动态被点击")
|
||||
debugInfoSync(" 动态ID: \(tappedMoment.dynamicId)")
|
||||
debugInfoSync(" 用户: \(tappedMoment.nick)")
|
||||
}
|
||||
)
|
||||
.padding(.leading, 16)
|
||||
.padding(.trailing, 32)
|
||||
.onAppear {
|
||||
// 当显示倒数第三个项目时,开始加载更多
|
||||
if index == viewModel.moments.count - 3 {
|
||||
viewModel.loadMoreData()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 加载更多状态指示器
|
||||
if viewModel.isLoadingMore {
|
||||
HStack {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
Text("加载更多...")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
.padding(.vertical, 20)
|
||||
}
|
||||
|
||||
// 没有更多数据提示
|
||||
if !viewModel.hasMore && !viewModel.moments.isEmpty {
|
||||
Text("没有更多数据了")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
.padding(.vertical, 20)
|
||||
}
|
||||
}
|
||||
.padding(.bottom, 160) // 为底部导航栏留出空间
|
||||
}
|
||||
}
|
||||
.refreshable {
|
||||
// 下拉刷新
|
||||
viewModel.refreshData()
|
||||
}
|
||||
.onAppear {
|
||||
// 调试信息
|
||||
debugInfoSync("📱 MomentListHomePage: 显示动态列表")
|
||||
debugInfoSync(" 动态数量: \(viewModel.moments.count)")
|
||||
debugInfoSync(" 是否有更多: \(viewModel.hasMore)")
|
||||
debugInfoSync(" 是否正在加载更多: \(viewModel.isLoadingMore)")
|
||||
}
|
||||
} else if viewModel.isLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.padding(.top, 20)
|
||||
} else if let error = viewModel.error {
|
||||
VStack(spacing: 16) {
|
||||
Text(error)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
.multilineTextAlignment(.center)
|
||||
.padding(.horizontal, 20)
|
||||
|
||||
// 重试按钮
|
||||
Button(action: {
|
||||
viewModel.refreshData()
|
||||
}) {
|
||||
Text("重试")
|
||||
.font(.system(size: 14, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.padding(.horizontal, 20)
|
||||
.padding(.vertical, 8)
|
||||
.background(Color.white.opacity(0.2))
|
||||
.cornerRadius(8)
|
||||
}
|
||||
}
|
||||
.padding(.top, 20)
|
||||
}
|
||||
|
||||
Spacer()
|
||||
}
|
||||
.safeAreaPadding(.top, 8)
|
||||
}
|
||||
.onAppear {
|
||||
viewModel.onAppear()
|
||||
}
|
||||
.onReceive(NotificationCenter.default.publisher(for: .init("CreateFeedPublished"))) { _ in
|
||||
viewModel.refreshData()
|
||||
}
|
||||
// MARK: - 图片预览弹窗(使用 sheet 以避免与发布页全屏弹窗冲突)
|
||||
.sheet(item: $previewItem) { item in
|
||||
ImagePreviewPager(
|
||||
images: item.images as [String],
|
||||
currentIndex: $previewCurrentIndex
|
||||
) {
|
||||
previewItem = nil
|
||||
debugInfoSync("📸 MomentListHomePage: 图片预览已关闭")
|
||||
}
|
||||
}
|
||||
// MARK: - 详情页弹窗
|
||||
.sheet(item: $selectedMoment) { moment in
|
||||
MomentDetailPage(moment: moment) {
|
||||
selectedMoment = nil
|
||||
debugInfoSync("📱 MomentListHomePage: 详情页已关闭")
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.presentationDetents([.large])
|
||||
.presentationDragIndicator(.visible)
|
||||
}
|
||||
// 发布页由上层统一控制
|
||||
}
|
||||
}
|
||||
421
yana/MVVM/View/MomentListItem.swift
Normal file
421
yana/MVVM/View/MomentListItem.swift
Normal file
@@ -0,0 +1,421 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - MomentListItem
|
||||
struct MomentListItem: View {
|
||||
let moment: MomentsInfo
|
||||
let onImageTap: (([String], Int)) -> Void // 新增:图片点击回调
|
||||
let onMomentTap: (MomentsInfo) -> Void // 新增:整体点击回调
|
||||
|
||||
// 新增:点赞相关状态
|
||||
@State private var isLikeLoading = false
|
||||
@State private var localIsLike: Bool
|
||||
@State private var localLikeCount: Int
|
||||
|
||||
init(
|
||||
moment: MomentsInfo,
|
||||
onImageTap: @escaping (([String], Int)) -> Void = { (arg) in let (_, _) = arg; },
|
||||
onMomentTap: @escaping (MomentsInfo) -> Void = { _ in }
|
||||
) {
|
||||
self.moment = moment
|
||||
self.onImageTap = onImageTap
|
||||
self.onMomentTap = onMomentTap
|
||||
// 初始化本地状态
|
||||
self._localIsLike = State(initialValue: moment.isLike)
|
||||
self._localLikeCount = State(initialValue: moment.likeCount)
|
||||
}
|
||||
|
||||
var body: some View {
|
||||
let isReviewing = moment.status == 0
|
||||
|
||||
ZStack(alignment: .bottomTrailing) {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 12)
|
||||
.fill(Color.clear)
|
||||
.overlay(
|
||||
RoundedRectangle(cornerRadius: 12)
|
||||
.stroke(Color.white.opacity(0.1), lineWidth: 1)
|
||||
)
|
||||
.shadow(color: Color(red: 0.43, green: 0.43, blue: 0.43, opacity: 0.34), radius: 10.7, x: 0, y: 1.9)
|
||||
|
||||
// 内容层
|
||||
VStack(alignment: .leading, spacing: 10) {
|
||||
// 用户信息
|
||||
HStack(alignment: .top) {
|
||||
// 头像
|
||||
CachedAsyncImage(url: moment.avatar) { image in
|
||||
image
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
} placeholder: {
|
||||
Circle()
|
||||
.fill(Color.gray.opacity(0.3))
|
||||
.overlay(
|
||||
Text(String(moment.nick.prefix(1)))
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
)
|
||||
}
|
||||
.frame(width: 40, height: 40)
|
||||
.clipShape(Circle())
|
||||
|
||||
VStack(alignment: .leading, spacing: 2) {
|
||||
Text(moment.nick)
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
UserIDDisplay(uid: moment.uid, fontSize: 12, textColor: .white.opacity(0.6))
|
||||
}
|
||||
Spacer()
|
||||
// 时间
|
||||
Text(formatDisplayTime(moment.publishTime))
|
||||
.font(.system(size: 12, weight: .bold))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
.padding(.horizontal, 6)
|
||||
.padding(.vertical, 2)
|
||||
.background(Color.white.opacity(0.15))
|
||||
.cornerRadius(4)
|
||||
}
|
||||
|
||||
// 动态内容
|
||||
if !moment.content.isEmpty {
|
||||
Text(moment.content)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.9))
|
||||
.multilineTextAlignment(.leading)
|
||||
.padding(.leading, 40 + 8) // 与用户名左边对齐
|
||||
}
|
||||
|
||||
// 图片网格
|
||||
if let images = moment.dynamicResList, !images.isEmpty {
|
||||
MomentImageGrid(
|
||||
images: images,
|
||||
onImageTap: onImageTap
|
||||
)
|
||||
.padding(.leading, 40 + 8)
|
||||
.padding(.bottom, images.count == 2 ? 30 : 0) // 两张图片时增加底部间距
|
||||
}
|
||||
|
||||
// 互动按钮
|
||||
HStack(spacing: 20) {
|
||||
// Like 按钮与用户名左侧对齐
|
||||
Button(action: {
|
||||
if !isLikeLoading && !isReviewing {
|
||||
handleLikeTap()
|
||||
}
|
||||
}) {
|
||||
HStack(spacing: 4) {
|
||||
if isLikeLoading {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: localIsLike ? .red : .white.opacity(0.8)))
|
||||
.scaleEffect(0.8)
|
||||
} else {
|
||||
Image(systemName: localIsLike ? "heart.fill" : "heart")
|
||||
.font(.system(size: 16))
|
||||
}
|
||||
Text("\(localLikeCount)")
|
||||
.font(.system(size: 14))
|
||||
}
|
||||
.foregroundColor(localIsLike ? .red : .white.opacity(0.8))
|
||||
}
|
||||
.disabled(isLikeLoading || isReviewing)
|
||||
.opacity(isReviewing ? 0.5 : 1.0)
|
||||
.padding(.leading, 40 + 8) // 与用户名左侧对齐(头像宽度 + 间距)
|
||||
|
||||
Spacer()
|
||||
|
||||
// 审核中状态角标 - 右侧对齐日期,垂直居中对齐点赞按钮
|
||||
if isReviewing {
|
||||
Text("reviewing")
|
||||
.font(.system(size: 12, weight: .semibold))
|
||||
.foregroundColor(.white)
|
||||
.padding(.horizontal, 10)
|
||||
.padding(.vertical, 6)
|
||||
.background(Color.orange.opacity(0.85))
|
||||
.clipShape(Capsule())
|
||||
}
|
||||
}
|
||||
.padding(.top, 8)
|
||||
}
|
||||
.padding(16)
|
||||
}
|
||||
.contentShape(Rectangle())
|
||||
.onTapGesture {
|
||||
onMomentTap(moment)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 点赞处理逻辑
|
||||
private func handleLikeTap() {
|
||||
Task {
|
||||
await performLikeRequest()
|
||||
}
|
||||
}
|
||||
|
||||
private func performLikeRequest() async {
|
||||
// 设置加载状态
|
||||
await MainActor.run {
|
||||
isLikeLoading = true
|
||||
}
|
||||
|
||||
do {
|
||||
// 获取当前用户ID
|
||||
guard let currentUserId = await UserInfoManager.getCurrentUserId(),
|
||||
let currentUserIdInt = Int(currentUserId) else {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: "无法获取用户信息,请重新登录")
|
||||
return
|
||||
}
|
||||
|
||||
// 确定请求参数
|
||||
let status = localIsLike ? 0 : 1 // 0: 取消点赞, 1: 点赞
|
||||
|
||||
// 创建 API 服务实例
|
||||
let apiService = LiveAPIService()
|
||||
|
||||
// 创建请求
|
||||
let request = LikeDynamicRequest(
|
||||
dynamicId: moment.dynamicId,
|
||||
uid: currentUserIdInt,
|
||||
status: status,
|
||||
likedUid: moment.uid,
|
||||
worldId: moment.worldId
|
||||
)
|
||||
|
||||
debugInfoSync("📡 MomentListItem: 发送点赞请求")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 当前状态: \(localIsLike)")
|
||||
debugInfoSync(" 请求状态: \(status)")
|
||||
|
||||
// 发起请求
|
||||
let response: LikeDynamicResponse = try await apiService.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
// 处理响应, 只需要判断 code
|
||||
if response.code == 200 {
|
||||
localIsLike = !localIsLike
|
||||
localLikeCount = localIsLike ? localLikeCount+1 : localLikeCount-1
|
||||
debugInfoSync("✅ MomentListItem: 点赞操作成功")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 新状态: \(localIsLike)")
|
||||
debugInfoSync(" 新数量: \(localLikeCount)")
|
||||
} else {
|
||||
let errorMessage = response.message.isEmpty ? "点赞失败,请重试" : response.message
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: errorMessage)
|
||||
debugErrorSync("❌ MomentListItem: 点赞操作失败")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(errorMessage)")
|
||||
}
|
||||
}
|
||||
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: error.localizedDescription)
|
||||
debugErrorSync("❌ MomentListItem: 点赞请求异常")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 时间显示逻辑
|
||||
private func formatDisplayTime(_ timestamp: Int) -> String {
|
||||
let date = Date(timeIntervalSince1970: TimeInterval(timestamp) / 1000.0)
|
||||
let formatter = DateFormatter()
|
||||
formatter.locale = Locale(identifier: "zh_CN")
|
||||
let now = Date()
|
||||
let interval = now.timeIntervalSince(date)
|
||||
let calendar = Calendar.current
|
||||
if calendar.isDateInToday(date) {
|
||||
if interval < 60 {
|
||||
return "刚刚"
|
||||
} else if interval < 3600 {
|
||||
return "\(Int(interval / 60))分钟前"
|
||||
} else {
|
||||
return "\(Int(interval / 3600))小时前"
|
||||
}
|
||||
} else {
|
||||
formatter.dateFormat = "MM/dd"
|
||||
return formatter.string(from: date)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 图片网格组件
|
||||
struct MomentImageGrid: View {
|
||||
let images: [MomentsPicture]
|
||||
let onImageTap: (([String], Int)) -> Void // 新增:图片点击回调
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
let availableWidth = max(geometry.size.width, 1)
|
||||
let spacing: CGFloat = 8
|
||||
if availableWidth < 10 {
|
||||
Color.clear.frame(height: 1)
|
||||
} else {
|
||||
switch images.count {
|
||||
case 1:
|
||||
let imageSize: CGFloat = min(availableWidth * 0.6, 200)
|
||||
HStack {
|
||||
Spacer()
|
||||
MomentSquareImageView(
|
||||
image: images[0],
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, 0))
|
||||
}
|
||||
)
|
||||
Spacer()
|
||||
}
|
||||
case 2:
|
||||
let imageSize: CGFloat = (availableWidth - spacing) / 2
|
||||
HStack(spacing: spacing) {
|
||||
MomentSquareImageView(
|
||||
image: images[0],
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, 0))
|
||||
}
|
||||
)
|
||||
MomentSquareImageView(
|
||||
image: images[1],
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, 1))
|
||||
}
|
||||
)
|
||||
}
|
||||
case 3:
|
||||
let imageSize: CGFloat = (availableWidth - spacing * 2) / 3
|
||||
HStack(spacing: spacing) {
|
||||
ForEach(Array(images.prefix(3).enumerated()), id: \.element.id) { index, image in
|
||||
MomentSquareImageView(
|
||||
image: image,
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, index))
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
default:
|
||||
let imageSize: CGFloat = (availableWidth - spacing * 2) / 3
|
||||
let columns = Array(repeating: GridItem(.fixed(imageSize), spacing: spacing), count: 3)
|
||||
LazyVGrid(columns: columns, spacing: spacing) {
|
||||
ForEach(Array(images.prefix(9).enumerated()), id: \.element.id) { index, image in
|
||||
MomentSquareImageView(
|
||||
image: image,
|
||||
size: imageSize,
|
||||
onTap: {
|
||||
let imageUrls = images.compactMap { $0.resUrl }
|
||||
onImageTap((imageUrls, index))
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
.frame(height: calculateGridHeight())
|
||||
}
|
||||
|
||||
private func calculateGridHeight() -> CGFloat {
|
||||
switch images.count {
|
||||
case 1:
|
||||
return 200
|
||||
case 2:
|
||||
return 120
|
||||
case 3:
|
||||
return 100
|
||||
case 4...6:
|
||||
return 216
|
||||
default:
|
||||
return 340
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 正方形图片视图组件
|
||||
struct MomentSquareImageView: View {
|
||||
let image: MomentsPicture
|
||||
let size: CGFloat
|
||||
let onTap: () -> Void // 新增:点击回调
|
||||
|
||||
var body: some View {
|
||||
let safeSize = size.isFinite && size > 0 ? size : 100
|
||||
Button(action: onTap) {
|
||||
CachedAsyncImage(url: image.resUrl ?? "") { imageView in
|
||||
imageView
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
} placeholder: {
|
||||
Rectangle()
|
||||
.fill(Color.gray.opacity(0.3))
|
||||
.overlay(
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white.opacity(0.6)))
|
||||
.scaleEffect(0.8)
|
||||
)
|
||||
}
|
||||
.frame(width: safeSize, height: safeSize)
|
||||
.clipped()
|
||||
.cornerRadius(8)
|
||||
}
|
||||
.buttonStyle(PlainButtonStyle()) // 使用PlainButtonStyle避免默认的按钮样式
|
||||
}
|
||||
}
|
||||
|
||||
//#Preview {
|
||||
// // 创建测试数据
|
||||
// let testMoment = MomentsInfo(
|
||||
// dynamicId: 1,
|
||||
// uid: 123456,
|
||||
// nick: "测试用户",
|
||||
// avatar: "",
|
||||
// type: 0,
|
||||
// content: "这是一条测试动态内容,用来测试 MomentListItem 的显示效果。",
|
||||
// likeCount: 42,
|
||||
// isLike: false,
|
||||
// commentCount: 5,
|
||||
// publishTime: Int(Date().timeIntervalSince1970 * 1000),
|
||||
// worldId: 1,
|
||||
// status: 1,
|
||||
// playCount: nil,
|
||||
// dynamicResList: [
|
||||
// MomentsPicture(id: 1, resUrl: "https://picsum.photos/300/300", format: "jpg", width: 300, height: 300, resDuration: nil),
|
||||
// MomentsPicture(id: 2, resUrl: "https://picsum.photos/301/301", format: "jpg", width: 301, height: 301, resDuration: nil)
|
||||
// ],
|
||||
// gender: nil,
|
||||
// squareTop: nil,
|
||||
// topicTop: nil,
|
||||
// newUser: nil,
|
||||
// defUser: nil,
|
||||
// scene: nil,
|
||||
// userVipInfoVO: nil,
|
||||
// headwearPic: nil,
|
||||
// headwearEffect: nil,
|
||||
// headwearType: nil,
|
||||
// headwearName: nil,
|
||||
// headwearId: nil,
|
||||
// experLevelPic: nil,
|
||||
// charmLevelPic: nil,
|
||||
// isCustomWord: nil,
|
||||
// labelList: nil
|
||||
// )
|
||||
//
|
||||
// MomentListItem(
|
||||
// moment: testMoment,
|
||||
// onImageTap: { images, index in
|
||||
// print("图片被点击: 索引 \(index), 图片数量 \(images.count)")
|
||||
// }
|
||||
// )
|
||||
// .padding()
|
||||
// .background(Color.black)
|
||||
//}
|
||||
123
yana/MVVM/View/NineGridImagePicker.swift
Normal file
123
yana/MVVM/View/NineGridImagePicker.swift
Normal file
@@ -0,0 +1,123 @@
|
||||
import SwiftUI
|
||||
import PhotosUI
|
||||
|
||||
struct NineGridImagePicker: View {
|
||||
@Binding var images: [UIImage]
|
||||
var maxCount: Int = 9
|
||||
var cornerRadius: CGFloat = 16
|
||||
var spacing: CGFloat = 8
|
||||
var horizontalPadding: CGFloat = 20
|
||||
var onTapImage: (Int) -> Void = { _ in }
|
||||
|
||||
@State private var pickerItems: [PhotosPickerItem] = []
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
let columns = Array(repeating: GridItem(.flexible(), spacing: spacing), count: 3)
|
||||
let columnsCount: CGFloat = 3
|
||||
let totalSpacing = spacing * (columnsCount - 1)
|
||||
let availableWidth = geometry.size.width - horizontalPadding * 2
|
||||
let cellSide = (availableWidth - totalSpacing) / columnsCount
|
||||
|
||||
LazyVGrid(columns: columns, spacing: spacing) {
|
||||
ForEach(0..<maxCount, id: \.self) { index in
|
||||
ZStack {
|
||||
// 占位背景(仅 DEBUG 可见)
|
||||
#if DEBUG
|
||||
if index >= images.count && !(index == images.count && images.count < maxCount) {
|
||||
RoundedRectangle(cornerRadius: cornerRadius)
|
||||
.fill(Color.white.opacity(0.08))
|
||||
}
|
||||
#endif
|
||||
|
||||
if index < images.count {
|
||||
// 图片格子
|
||||
ZStack(alignment: .topTrailing) {
|
||||
Image(uiImage: images[index])
|
||||
.resizable()
|
||||
.scaledToFill()
|
||||
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
||||
.contentShape(RoundedRectangle(cornerRadius: cornerRadius))
|
||||
.onTapGesture { onTapImage(index) }
|
||||
|
||||
Button {
|
||||
removeImage(at: index)
|
||||
} label: {
|
||||
Image(systemName: "xmark.circle.fill")
|
||||
.foregroundColor(.white)
|
||||
.background(Circle().fill(Color.black.opacity(0.4)))
|
||||
.font(.system(size: 16, weight: .bold))
|
||||
}
|
||||
.padding(6)
|
||||
.buttonStyle(.plain)
|
||||
}
|
||||
} else if index == images.count && images.count < maxCount {
|
||||
// 添加按钮格子
|
||||
PhotosPicker(
|
||||
selection: $pickerItems,
|
||||
maxSelectionCount: maxCount - images.count,
|
||||
selectionBehavior: .ordered,
|
||||
matching: .images
|
||||
) {
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: cornerRadius)
|
||||
.fill(Color(hex: 0x1C143A))
|
||||
Image(systemName: "plus")
|
||||
.foregroundColor(.white.opacity(0.6))
|
||||
.font(.system(size: 32, weight: .semibold))
|
||||
}
|
||||
}
|
||||
.onChange(of: pickerItems) { _, newItems in
|
||||
handlePickerItems(newItems)
|
||||
}
|
||||
}
|
||||
}
|
||||
.frame(height: cellSide)
|
||||
.clipShape(RoundedRectangle(cornerRadius: cornerRadius))
|
||||
.contentShape(RoundedRectangle(cornerRadius: cornerRadius))
|
||||
}
|
||||
}
|
||||
.padding(.horizontal, horizontalPadding)
|
||||
}
|
||||
.frame(height: gridHeight(forCount: max(images.count, 1)))
|
||||
}
|
||||
|
||||
private func gridHeight(forCount count: Int) -> CGFloat {
|
||||
// 通过一个近似:用屏幕宽度估算高度以确保父布局正确测量。
|
||||
// 每行 3 个,行数 = ceil(count / 3.0)。在 GeometryReader 中真实高度会覆盖此近似。
|
||||
let screenWidth = UIScreen.main.bounds.width
|
||||
let columnsCount: CGFloat = 3
|
||||
let totalSpacing = spacing * (columnsCount - 1)
|
||||
let availableWidth = screenWidth - horizontalPadding * 2
|
||||
let side = (availableWidth - totalSpacing) / columnsCount
|
||||
let rows = ceil(CGFloat(count) / 3.0)
|
||||
let totalRowSpacing = spacing * max(rows - 1, 0)
|
||||
return side * rows + totalRowSpacing
|
||||
}
|
||||
|
||||
private func handlePickerItems(_ items: [PhotosPickerItem]) {
|
||||
guard !items.isEmpty else { return }
|
||||
Task { @MainActor in
|
||||
var appended: [UIImage] = []
|
||||
for item in items {
|
||||
if images.count + appended.count >= maxCount { break }
|
||||
if let data = try? await item.loadTransferable(type: Data.self),
|
||||
let image = UIImage(data: data) {
|
||||
appended.append(image)
|
||||
}
|
||||
}
|
||||
if !appended.isEmpty {
|
||||
images.append(contentsOf: appended)
|
||||
}
|
||||
pickerItems = []
|
||||
}
|
||||
}
|
||||
|
||||
private func removeImage(at index: Int) {
|
||||
guard images.indices.contains(index) else { return }
|
||||
images.remove(at: index)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
358
yana/MVVM/View/SettingPage.swift
Normal file
358
yana/MVVM/View/SettingPage.swift
Normal file
@@ -0,0 +1,358 @@
|
||||
import SwiftUI
|
||||
import PhotosUI
|
||||
import UIKit
|
||||
|
||||
// MARK: - Setting Page
|
||||
|
||||
struct SettingPage: View {
|
||||
@StateObject private var viewModel = SettingViewModel()
|
||||
let onBack: () -> Void
|
||||
let onLogout: () -> Void
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
ZStack {
|
||||
// 背景颜色
|
||||
Color(hex: 0x0C0527)
|
||||
.ignoresSafeArea(.all)
|
||||
|
||||
VStack(spacing: 0) {
|
||||
// 顶部导航栏
|
||||
HStack {
|
||||
Button(action: {
|
||||
viewModel.onBackTapped()
|
||||
}) {
|
||||
Image(systemName: "chevron.left")
|
||||
.font(.system(size: 24, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.frame(width: 44, height: 44)
|
||||
}
|
||||
|
||||
Spacer()
|
||||
|
||||
Text(LocalizedString("appSetting.title", comment: "编辑"))
|
||||
.font(.system(size: 18, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
|
||||
Spacer()
|
||||
|
||||
// 占位,保持标题居中
|
||||
Color.clear
|
||||
.frame(width: 44, height: 44)
|
||||
}
|
||||
.padding(.horizontal, 16)
|
||||
.padding(.top, 8)
|
||||
|
||||
// 主要内容区域
|
||||
ScrollView {
|
||||
VStack(spacing: 0) {
|
||||
// 头像设置区域
|
||||
avatarSection()
|
||||
.padding(.top, 20)
|
||||
|
||||
// 个人信息设置区域
|
||||
personalInfoSection()
|
||||
.padding(.top, 30)
|
||||
|
||||
// 其他设置区域
|
||||
otherSettingsSection()
|
||||
.padding(.top, 20)
|
||||
|
||||
Spacer(minLength: 40)
|
||||
|
||||
// 退出登录按钮
|
||||
logoutSection()
|
||||
.padding(.bottom, 40)
|
||||
}
|
||||
.padding(.horizontal, 20)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
.navigationBarHidden(true)
|
||||
.onAppear {
|
||||
viewModel.onBack = onBack
|
||||
viewModel.onLogout = onLogout
|
||||
viewModel.onAppear()
|
||||
}
|
||||
// 图片源选择 ActionSheet
|
||||
.confirmationDialog(
|
||||
"请选择图片来源",
|
||||
isPresented: $viewModel.showImageSourceActionSheet,
|
||||
titleVisibility: .visible
|
||||
) {
|
||||
Button(LocalizedString("app_settings.take_photo", comment: "拍照")) {
|
||||
viewModel.selectImageSource(.camera)
|
||||
}
|
||||
Button(LocalizedString("app_settings.select_from_album", comment: "从相册选择")) {
|
||||
viewModel.selectImageSource(.photoLibrary)
|
||||
}
|
||||
Button(LocalizedString("common.cancel", comment: "取消"), role: .cancel) { }
|
||||
}
|
||||
// 相机选择器
|
||||
.sheet(isPresented: $viewModel.showCamera) {
|
||||
CameraPicker { image in
|
||||
guard let image = image else {
|
||||
return
|
||||
}
|
||||
viewModel.onCameraImagePicked(image)
|
||||
}
|
||||
}
|
||||
// 相册选择器
|
||||
.photosPicker(
|
||||
isPresented: $viewModel.showPhotoPicker,
|
||||
selection: $viewModel.selectedPhotoItems,
|
||||
maxSelectionCount: 1,
|
||||
matching: .images
|
||||
)
|
||||
// 昵称编辑弹窗
|
||||
.alert(LocalizedString("appSetting.nickname", comment: "编辑昵称"), isPresented: $viewModel.isEditingNickname) {
|
||||
TextField(LocalizedString("appSetting.nickname", comment: "请输入昵称"), text: $viewModel.nicknameInput)
|
||||
.onChange(of: viewModel.nicknameInput) { _, newValue in
|
||||
viewModel.onNicknameInputChanged(newValue)
|
||||
}
|
||||
Button(LocalizedString("common.cancel", comment: "取消")) {
|
||||
viewModel.isEditingNickname = false
|
||||
}
|
||||
Button(LocalizedString("common.confirm", comment: "确认")) {
|
||||
viewModel.onNicknameEditConfirmed()
|
||||
}
|
||||
} message: {
|
||||
Text(LocalizedString("appSetting.nickname", comment: "请输入新的昵称"))
|
||||
}
|
||||
// 登出确认弹窗
|
||||
.alert(LocalizedString("appSetting.logoutConfirmation.title", comment: "确认退出"), isPresented: $viewModel.showLogoutConfirmation) {
|
||||
Button(LocalizedString("common.cancel", comment: "取消"), role: .cancel) {
|
||||
viewModel.showLogoutConfirmation = false
|
||||
}
|
||||
Button(LocalizedString("appSetting.logoutConfirmation.confirm", comment: "确认退出"), role: .destructive) {
|
||||
viewModel.onLogoutConfirmed()
|
||||
viewModel.showLogoutConfirmation = false
|
||||
}
|
||||
} message: {
|
||||
Text(LocalizedString("appSetting.logoutConfirmation.message", comment: "确定要退出当前账户吗?"))
|
||||
}
|
||||
// 关于我们弹窗
|
||||
.alert(LocalizedString("appSetting.aboutUs.title", comment: "关于我们"), isPresented: $viewModel.showAboutUs) {
|
||||
Button(LocalizedString("common.ok", comment: "确定")) {
|
||||
viewModel.showAboutUs = false
|
||||
}
|
||||
} message: {
|
||||
VStack(alignment: .leading, spacing: 8) {
|
||||
Text(LocalizedString("feedList.title", comment: "享受您的生活时光"))
|
||||
.font(.headline)
|
||||
Text(LocalizedString("feedList.slogan", comment: "疾病如同残酷的统治者,\n而时间是我们最宝贵的财富。\n我们活着的每一刻,都是对不可避免命运的胜利。"))
|
||||
.font(.body)
|
||||
}
|
||||
}
|
||||
// WebView 导航
|
||||
.webView(
|
||||
isPresented: $viewModel.showPrivacyPolicy,
|
||||
url: APIConfiguration.webURL(for: .privacyPolicy)
|
||||
)
|
||||
.onChange(of: viewModel.showPrivacyPolicy) { _, isPresented in
|
||||
if !isPresented {
|
||||
viewModel.onPrivacyPolicyDismissed()
|
||||
}
|
||||
}
|
||||
.webView(
|
||||
isPresented: $viewModel.showUserAgreement,
|
||||
url: APIConfiguration.webURL(for: .userAgreement)
|
||||
)
|
||||
.onChange(of: viewModel.showUserAgreement) { _, isPresented in
|
||||
if !isPresented {
|
||||
viewModel.onUserAgreementDismissed()
|
||||
}
|
||||
}
|
||||
.webView(
|
||||
isPresented: $viewModel.showDeactivateAccount,
|
||||
url: APIConfiguration.webURL(for: .deactivateAccount)
|
||||
)
|
||||
.onChange(of: viewModel.showDeactivateAccount) { _, isPresented in
|
||||
if !isPresented {
|
||||
viewModel.onDeactivateAccountDismissed()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 头像设置区域
|
||||
@ViewBuilder
|
||||
private func avatarSection() -> some View {
|
||||
VStack(spacing: 16) {
|
||||
// 头像
|
||||
Button(action: {
|
||||
viewModel.onAvatarTapped()
|
||||
}) {
|
||||
ZStack {
|
||||
AsyncImage(url: URL(string: viewModel.userInfo?.avatar ?? "")) { image in
|
||||
image
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
} placeholder: {
|
||||
Image(systemName: "person.circle.fill")
|
||||
.resizable()
|
||||
.aspectRatio(contentMode: .fill)
|
||||
.foregroundColor(.gray)
|
||||
}
|
||||
.frame(width: 120, height: 120)
|
||||
.clipShape(Circle())
|
||||
|
||||
// 相机图标覆盖
|
||||
VStack {
|
||||
Spacer()
|
||||
HStack {
|
||||
Spacer()
|
||||
Circle()
|
||||
.fill(Color.purple)
|
||||
.frame(width: 32, height: 32)
|
||||
.overlay(
|
||||
Image(systemName: "camera")
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
)
|
||||
}
|
||||
}
|
||||
.frame(width: 120, height: 120)
|
||||
}
|
||||
}
|
||||
.disabled(viewModel.isUploadingAvatar || viewModel.isUpdatingUser)
|
||||
|
||||
// 上传状态提示
|
||||
if viewModel.isUploadingAvatar {
|
||||
Text("正在上传头像...")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
|
||||
if let error = viewModel.avatarUploadError {
|
||||
Text(error)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 个人信息设置区域
|
||||
@ViewBuilder
|
||||
private func personalInfoSection() -> some View {
|
||||
VStack(spacing: 0) {
|
||||
// 昵称设置
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.nickname", comment: "昵称"),
|
||||
subtitle: viewModel.userInfo?.nick ?? LocalizedString("app_settings.not_set", comment: "未设置"),
|
||||
action: {
|
||||
viewModel.onNicknameTapped()
|
||||
}
|
||||
)
|
||||
.disabled(viewModel.isUpdatingUser)
|
||||
|
||||
// 更新状态提示
|
||||
if viewModel.isUpdatingUser {
|
||||
HStack {
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(0.8)
|
||||
Text("正在更新...")
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.white.opacity(0.8))
|
||||
}
|
||||
.padding(.top, 8)
|
||||
}
|
||||
|
||||
if let error = viewModel.updateUserError {
|
||||
Text(error)
|
||||
.font(.system(size: 14))
|
||||
.foregroundColor(.red)
|
||||
.padding(.top, 8)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 其他设置区域
|
||||
@ViewBuilder
|
||||
private func otherSettingsSection() -> some View {
|
||||
VStack(spacing: 0) {
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.personalInfoPermissions", comment: "个人信息与权限"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onPersonalInfoPermissionsTapped() }
|
||||
)
|
||||
|
||||
Divider()
|
||||
.background(Color.white.opacity(0.2))
|
||||
.padding(.leading, 16)
|
||||
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.help", comment: "帮助"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onHelpTapped() }
|
||||
)
|
||||
|
||||
Divider()
|
||||
.background(Color.white.opacity(0.2))
|
||||
.padding(.leading, 16)
|
||||
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.clearCache", comment: "清除缓存"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onClearCacheTapped() }
|
||||
)
|
||||
|
||||
Divider()
|
||||
.background(Color.white.opacity(0.2))
|
||||
.padding(.leading, 16)
|
||||
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.checkUpdates", comment: "检查更新"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onCheckUpdatesTapped() }
|
||||
)
|
||||
|
||||
Divider()
|
||||
.background(Color.white.opacity(0.2))
|
||||
.padding(.leading, 16)
|
||||
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.deactivateAccount", comment: "注销账号"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onDeactivateAccountTapped() }
|
||||
)
|
||||
|
||||
Divider()
|
||||
.background(Color.white.opacity(0.2))
|
||||
.padding(.leading, 16)
|
||||
|
||||
SettingRow(
|
||||
title: LocalizedString("appSetting.aboutUs", comment: "关于我们"),
|
||||
subtitle: "",
|
||||
action: { viewModel.onAboutUsTapped() }
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 退出登录区域
|
||||
@ViewBuilder
|
||||
private func logoutSection() -> some View {
|
||||
VStack(spacing: 12) {
|
||||
// 退出登录按钮
|
||||
Button(action: {
|
||||
viewModel.onLogoutTapped()
|
||||
}) {
|
||||
Text(LocalizedString("appSetting.logoutAccount", comment: "退出账户"))
|
||||
.font(.system(size: 16, weight: .medium))
|
||||
.foregroundColor(.white)
|
||||
.frame(maxWidth: .infinity)
|
||||
.padding(.vertical, 16)
|
||||
.background(Color.red.opacity(0.8))
|
||||
.cornerRadius(12)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//#Preview {
|
||||
// SettingPage(
|
||||
// onBack: {},
|
||||
// onLogout: {}
|
||||
// )
|
||||
//}
|
||||
194
yana/MVVM/ViewModel/IDLoginViewModel.swift
Normal file
194
yana/MVVM/ViewModel/IDLoginViewModel.swift
Normal file
@@ -0,0 +1,194 @@
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
// MARK: - IDLogin ViewModel
|
||||
|
||||
@MainActor
|
||||
class IDLoginViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var userID: String = ""
|
||||
@Published var password: String = ""
|
||||
@Published var isPasswordVisible: Bool = false
|
||||
@Published var isLoading: Bool = false
|
||||
@Published var errorMessage: String?
|
||||
@Published var showRecoverPassword: Bool = false
|
||||
@Published var loginStep: LoginStep = .input
|
||||
|
||||
// MARK: - Ticket 相关状态
|
||||
@Published var isTicketLoading: Bool = false
|
||||
@Published var ticketError: String?
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onBack: (() -> Void)?
|
||||
var onLoginSuccess: (() -> Void)?
|
||||
|
||||
// MARK: - Private Properties
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
|
||||
// MARK: - Enums
|
||||
enum LoginStep: Equatable {
|
||||
case input // 初始状态
|
||||
case authenticating // 正在进行 OAuth 认证
|
||||
case gettingTicket // 正在获取 Ticket
|
||||
case completed // 认证完成
|
||||
case failed // 认证失败
|
||||
}
|
||||
|
||||
// MARK: - Computed Properties
|
||||
var isLoginButtonEnabled: Bool {
|
||||
return !isLoading && !userID.isEmpty && !password.isEmpty
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onBackTapped() {
|
||||
onBack?()
|
||||
}
|
||||
|
||||
func onLoginTapped() {
|
||||
guard isLoginButtonEnabled else { return }
|
||||
|
||||
isLoading = true
|
||||
errorMessage = nil
|
||||
ticketError = nil
|
||||
loginStep = .authenticating
|
||||
|
||||
Task {
|
||||
do {
|
||||
let result = try await performLogin()
|
||||
await MainActor.run {
|
||||
self.handleLoginResult(result)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleLoginError(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func onRecoverPasswordTapped() {
|
||||
showRecoverPassword = true
|
||||
}
|
||||
|
||||
func onRecoverPasswordBack() {
|
||||
showRecoverPassword = false
|
||||
}
|
||||
|
||||
// MARK: - Private Methods
|
||||
private func performLogin() async throws -> Bool {
|
||||
// 第一步:OAuth认证
|
||||
let accountModel = try await performOAuthAuthentication()
|
||||
|
||||
// 第二步:获取Ticket
|
||||
let completeAccountModel = try await performTicketRequest(accountModel: accountModel)
|
||||
|
||||
// 第三步:保存完整的AccountModel
|
||||
await UserInfoManager.saveAccountModel(completeAccountModel)
|
||||
|
||||
// 第四步:获取用户信息(如果API没有返回)
|
||||
await fetchUserInfoIfNeeded(accountModel: completeAccountModel)
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
// MARK: - OAuth认证
|
||||
private func performOAuthAuthentication() async throws -> AccountModel {
|
||||
// 使用LoginHelper创建登录请求(包含DES加密)
|
||||
guard let loginRequest = await LoginHelper.createIDLoginRequest(
|
||||
userID: userID,
|
||||
password: password
|
||||
) else {
|
||||
throw APIError.custom("DES加密失败")
|
||||
}
|
||||
|
||||
let apiService = LiveAPIService()
|
||||
let response: IDLoginResponse = try await apiService.request(loginRequest)
|
||||
|
||||
if response.code == 200, let data = response.data {
|
||||
// 保存用户信息(如果API返回了用户信息)
|
||||
if let userInfo = data.userInfo {
|
||||
await UserInfoManager.saveUserInfo(userInfo)
|
||||
}
|
||||
|
||||
// 创建账户模型(此时ticket为空)
|
||||
guard let accountModel = AccountModel.from(loginData: data) else {
|
||||
throw APIError.custom("账户信息无效")
|
||||
}
|
||||
|
||||
return accountModel
|
||||
} else {
|
||||
throw APIError.custom(response.message ?? "Login failed")
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Ticket获取
|
||||
private func performTicketRequest(accountModel: AccountModel) async throws -> AccountModel {
|
||||
await MainActor.run {
|
||||
self.isTicketLoading = true
|
||||
self.ticketError = nil
|
||||
self.loginStep = .gettingTicket
|
||||
}
|
||||
|
||||
let apiService = LiveAPIService()
|
||||
|
||||
// 创建ticket请求
|
||||
let ticketRequest = TicketHelper.createTicketRequest(
|
||||
accessToken: accountModel.accessToken ?? "",
|
||||
uid: accountModel.uid.flatMap { Int($0) }
|
||||
)
|
||||
|
||||
let ticketResponse: TicketResponse = try await apiService.request(ticketRequest)
|
||||
|
||||
await MainActor.run {
|
||||
self.isTicketLoading = false
|
||||
}
|
||||
|
||||
if ticketResponse.isSuccess {
|
||||
if let ticket = ticketResponse.ticket {
|
||||
debugInfoSync("✅ Ticket 获取成功: \(ticket)")
|
||||
|
||||
// 更新AccountModel,添加ticket
|
||||
let completeAccountModel = accountModel.withTicket(ticket)
|
||||
return completeAccountModel
|
||||
} else {
|
||||
throw APIError.custom("Ticket为空")
|
||||
}
|
||||
} else {
|
||||
throw APIError.custom(ticketResponse.errorMessage)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 用户信息获取
|
||||
private func fetchUserInfoIfNeeded(accountModel: AccountModel) async {
|
||||
// 如果API没有返回用户信息,则从服务器获取
|
||||
let apiService = LiveAPIService()
|
||||
if let userInfo = await UserInfoManager.fetchUserInfoFromServer(
|
||||
uid: accountModel.uid,
|
||||
apiService: apiService
|
||||
) {
|
||||
await UserInfoManager.saveUserInfo(userInfo)
|
||||
debugInfoSync("✅ 用户信息获取成功")
|
||||
} else {
|
||||
debugErrorSync("❌ 用户信息获取失败,但不影响登录流程")
|
||||
}
|
||||
}
|
||||
|
||||
private func handleLoginResult(_ success: Bool) {
|
||||
isLoading = false
|
||||
isTicketLoading = false
|
||||
if success {
|
||||
loginStep = .completed
|
||||
debugInfoSync("✅ ID 登录完整流程成功")
|
||||
onLoginSuccess?()
|
||||
}
|
||||
}
|
||||
|
||||
private func handleLoginError(_ error: Error) {
|
||||
isLoading = false
|
||||
isTicketLoading = false
|
||||
errorMessage = error.localizedDescription
|
||||
loginStep = .failed
|
||||
debugErrorSync("❌ ID 登录失败: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
|
||||
64
yana/MVVM/ViewModel/MainViewModel.swift
Normal file
64
yana/MVVM/ViewModel/MainViewModel.swift
Normal file
@@ -0,0 +1,64 @@
|
||||
import SwiftUI
|
||||
|
||||
// MARK: - Main ViewModel
|
||||
|
||||
@MainActor
|
||||
class MainViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var selectedTab: Tab = .feed
|
||||
@Published var isLoggedOut: Bool = false
|
||||
@Published var navigationPath = NavigationPath()
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onLogout: (() -> Void)?
|
||||
var onAddButtonTapped: (() -> Void)?
|
||||
|
||||
// MARK: - Enums
|
||||
enum Tab: String, CaseIterable {
|
||||
case feed = "feed"
|
||||
case me = "me"
|
||||
|
||||
var title: String {
|
||||
switch self {
|
||||
case .feed:
|
||||
return "Feed"
|
||||
case .me:
|
||||
return "Me"
|
||||
}
|
||||
}
|
||||
|
||||
var iconName: String {
|
||||
switch self {
|
||||
case .feed:
|
||||
return "list.bullet"
|
||||
case .me:
|
||||
return "person.circle"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onAppear() {
|
||||
debugInfoSync("🚀 MainView onAppear")
|
||||
debugInfoSync(" 当前selectedTab: \(selectedTab)")
|
||||
}
|
||||
|
||||
func onTabChanged(_ newTab: Tab) {
|
||||
selectedTab = newTab
|
||||
debugInfoSync("🔄 MainView selectedTab changed: \(newTab)")
|
||||
}
|
||||
|
||||
func onLogoutTapped() {
|
||||
isLoggedOut = true
|
||||
onLogout?()
|
||||
}
|
||||
|
||||
func onTopRightButtonTapped() {
|
||||
switch selectedTab {
|
||||
case .feed:
|
||||
navigationPath.append(AppRoute.publish)
|
||||
case .me:
|
||||
navigationPath.append(AppRoute.setting)
|
||||
}
|
||||
}
|
||||
}
|
||||
88
yana/MVVM/ViewModel/MePageViewModel.swift
Normal file
88
yana/MVVM/ViewModel/MePageViewModel.swift
Normal file
@@ -0,0 +1,88 @@
|
||||
import Foundation
|
||||
import SwiftUI
|
||||
|
||||
@MainActor
|
||||
final class MePageViewModel: ObservableObject {
|
||||
@Published var userId: Int = 0
|
||||
@Published var nickname: String = ""
|
||||
@Published var avatarURL: String = ""
|
||||
|
||||
@Published var moments: [MomentsInfo] = []
|
||||
@Published var isLoading: Bool = false
|
||||
@Published var isLoadingMore: Bool = false
|
||||
@Published var errorMessage: String? = nil
|
||||
@Published var hasMore: Bool = true
|
||||
|
||||
private var page: Int = 1
|
||||
private let pageSize: Int = 20
|
||||
|
||||
func onAppear() {
|
||||
Task { @MainActor in
|
||||
await loadCurrentUser()
|
||||
// 仅首次或空列表时加载,避免每次 Tab 切换重复请求
|
||||
if moments.isEmpty {
|
||||
await refreshData()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func refreshData() async {
|
||||
page = 1
|
||||
hasMore = true
|
||||
errorMessage = nil
|
||||
isLoading = true
|
||||
moments.removeAll()
|
||||
defer { isLoading = false }
|
||||
await fetchMyMoments(page: page)
|
||||
}
|
||||
|
||||
func loadMoreData() {
|
||||
guard !isLoadingMore, hasMore else { return }
|
||||
isLoadingMore = true
|
||||
Task { @MainActor in
|
||||
defer { isLoadingMore = false }
|
||||
page += 1
|
||||
await fetchMyMoments(page: page)
|
||||
}
|
||||
}
|
||||
|
||||
private func loadCurrentUser() async {
|
||||
// 从缓存/Keychain 获取当前登录用户信息
|
||||
if let account = await UserInfoManager.getAccountModel() {
|
||||
if let uidString = account.uid, let uid = Int(uidString) {
|
||||
userId = uid
|
||||
}
|
||||
// 优先从缓存的 UserInfo 获取更完整的信息
|
||||
if let info = await UserInfoManager.getUserInfo() {
|
||||
nickname = info.nick ?? nickname
|
||||
avatarURL = info.avatar ?? avatarURL
|
||||
}
|
||||
}
|
||||
// 兜底
|
||||
if nickname.isEmpty { nickname = "未知用户" }
|
||||
}
|
||||
|
||||
private func fetchMyMoments(page: Int) async {
|
||||
guard userId > 0 else {
|
||||
errorMessage = "未登录或用户ID无效"
|
||||
return
|
||||
}
|
||||
let api: any APIServiceProtocol & Sendable = LiveAPIService()
|
||||
let request = GetMyDynamicRequest(fromUid: userId, uid: userId, page: page, pageSize: pageSize)
|
||||
do {
|
||||
let response = try await api.request(request)
|
||||
if let list = response.data {
|
||||
let items = list.map { $0.toMomentsInfo() }
|
||||
if items.isEmpty { hasMore = false }
|
||||
moments.append(contentsOf: items)
|
||||
} else {
|
||||
hasMore = false
|
||||
}
|
||||
} catch {
|
||||
errorMessage = error.localizedDescription
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
114
yana/MVVM/ViewModel/MomentDetailViewModel.swift
Normal file
114
yana/MVVM/ViewModel/MomentDetailViewModel.swift
Normal file
@@ -0,0 +1,114 @@
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
// MARK: - MomentDetailViewModel
|
||||
|
||||
@MainActor
|
||||
final class MomentDetailViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var moment: MomentsInfo
|
||||
@Published var isLikeLoading = false
|
||||
@Published var localIsLike: Bool
|
||||
@Published var localLikeCount: Int
|
||||
@Published var showImagePreview = false
|
||||
@Published var images: [String] = []
|
||||
@Published var currentIndex: Int = 0
|
||||
|
||||
// MARK: - Private Properties
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
|
||||
// MARK: - Initialization
|
||||
init(moment: MomentsInfo) {
|
||||
self.moment = moment
|
||||
self.localIsLike = moment.isLike
|
||||
self.localLikeCount = moment.likeCount
|
||||
self.images = moment.dynamicResList?.compactMap { $0.resUrl } ?? []
|
||||
|
||||
debugInfoSync("📱 MomentDetailViewModel: 初始化")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 用户: \(moment.nick)")
|
||||
debugInfoSync(" 图片数量: \(images.count)")
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onImageTap(_ index: Int) {
|
||||
currentIndex = index
|
||||
showImagePreview = true
|
||||
debugInfoSync("📸 MomentDetailViewModel: 图片被点击,索引: \(index)")
|
||||
}
|
||||
|
||||
func like() {
|
||||
guard !isLikeLoading, moment.status != 0 else {
|
||||
debugInfoSync("⏸️ MomentDetailViewModel: 跳过点赞 - 正在加载: \(isLikeLoading), 审核中: \(moment.status == 0)")
|
||||
return
|
||||
}
|
||||
|
||||
isLikeLoading = true
|
||||
debugInfoSync("📡 MomentDetailViewModel: 开始点赞操作")
|
||||
|
||||
Task {
|
||||
do {
|
||||
// 获取当前用户ID
|
||||
guard let uidStr = await UserInfoManager.getCurrentUserId(),
|
||||
let uid = Int(uidStr) else {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: "无法获取用户信息,请重新登录")
|
||||
return
|
||||
}
|
||||
|
||||
// 确定请求参数
|
||||
let status = localIsLike ? 0 : 1 // 0: 取消点赞, 1: 点赞
|
||||
|
||||
// 创建 API 服务实例
|
||||
let api = LiveAPIService()
|
||||
|
||||
// 创建请求
|
||||
let request = LikeDynamicRequest(
|
||||
dynamicId: moment.dynamicId,
|
||||
uid: uid,
|
||||
status: status,
|
||||
likedUid: moment.uid,
|
||||
worldId: moment.worldId
|
||||
)
|
||||
|
||||
debugInfoSync("📡 MomentDetailViewModel: 发送点赞请求")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 当前状态: \(localIsLike)")
|
||||
debugInfoSync(" 请求状态: \(status)")
|
||||
|
||||
// 发起请求
|
||||
let response: LikeDynamicResponse = try await api.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
// 处理响应
|
||||
if response.code == 200 {
|
||||
localIsLike.toggle()
|
||||
localLikeCount += localIsLike ? 1 : -1
|
||||
debugInfoSync("✅ MomentDetailViewModel: 点赞操作成功")
|
||||
debugInfoSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugInfoSync(" 新状态: \(localIsLike)")
|
||||
debugInfoSync(" 新数量: \(localLikeCount)")
|
||||
} else {
|
||||
let errorMessage = response.message.isEmpty ? "点赞失败,请重试" : response.message
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: errorMessage)
|
||||
debugErrorSync("❌ MomentDetailViewModel: 点赞操作失败")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(errorMessage)")
|
||||
}
|
||||
}
|
||||
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
isLikeLoading = false
|
||||
}
|
||||
setAPILoadingErrorSync(UUID(), errorMessage: error.localizedDescription)
|
||||
debugErrorSync("❌ MomentDetailViewModel: 点赞请求异常")
|
||||
debugErrorSync(" 动态ID: \(moment.dynamicId)")
|
||||
debugErrorSync(" 错误: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
171
yana/MVVM/ViewModel/MomentListHomeViewModel.swift
Normal file
171
yana/MVVM/ViewModel/MomentListHomeViewModel.swift
Normal file
@@ -0,0 +1,171 @@
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
// MARK: - MomentListHome ViewModel
|
||||
|
||||
@MainActor
|
||||
class MomentListHomeViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var isLoading: Bool = false
|
||||
@Published var error: String? = nil
|
||||
@Published var moments: [MomentsInfo] = []
|
||||
@Published var isLoaded: Bool = false
|
||||
|
||||
// MARK: - 分页相关属性
|
||||
@Published var isLoadingMore: Bool = false
|
||||
@Published var hasMore: Bool = true
|
||||
@Published var nextDynamicId: Int = 0
|
||||
|
||||
// MARK: - Private Properties
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onAppear() {
|
||||
debugInfoSync("📱 MomentListHomeViewModel onAppear")
|
||||
guard !isLoaded else {
|
||||
debugInfoSync("✅ MomentListHomeViewModel: 数据已加载,跳过重复请求")
|
||||
return
|
||||
}
|
||||
fetchLatestDynamics(isRefresh: true)
|
||||
}
|
||||
|
||||
// MARK: - 刷新数据
|
||||
func refreshData() {
|
||||
debugInfoSync("🔄 MomentListHomeViewModel: 开始刷新数据")
|
||||
fetchLatestDynamics(isRefresh: true)
|
||||
}
|
||||
|
||||
// MARK: - 加载更多数据
|
||||
func loadMoreData() {
|
||||
guard hasMore && !isLoadingMore && !isLoading else {
|
||||
debugInfoSync("⏸️ MomentListHomeViewModel: 跳过加载更多 - hasMore: \(hasMore), isLoadingMore: \(isLoadingMore), isLoading: \(isLoading)")
|
||||
return
|
||||
}
|
||||
debugInfoSync("📥 MomentListHomeViewModel: 开始加载更多数据")
|
||||
fetchLatestDynamics(isRefresh: false)
|
||||
}
|
||||
|
||||
// MARK: - Private Methods
|
||||
private func fetchLatestDynamics(isRefresh: Bool) {
|
||||
if isRefresh {
|
||||
isLoading = true
|
||||
error = nil
|
||||
debugInfoSync("🔄 MomentListHomeViewModel: 开始获取最新动态")
|
||||
} else {
|
||||
isLoadingMore = true
|
||||
debugInfoSync("📥 MomentListHomeViewModel: 开始加载更多动态")
|
||||
}
|
||||
|
||||
Task {
|
||||
// 检查认证信息
|
||||
let accountModel = await UserInfoManager.getAccountModel()
|
||||
if accountModel?.uid != nil {
|
||||
debugInfoSync("✅ MomentListHomeViewModel: 认证信息已准备好,开始获取动态")
|
||||
await performAPICall(isRefresh: isRefresh)
|
||||
} else {
|
||||
debugInfoSync("⏳ MomentListHomeViewModel: 认证信息未准备好,等待...")
|
||||
// 增加等待时间和重试次数
|
||||
for attempt in 1...3 {
|
||||
try? await Task.sleep(nanoseconds: 500_000_000) // 0.5秒
|
||||
let retryAccountModel = await UserInfoManager.getAccountModel()
|
||||
if retryAccountModel?.uid != nil {
|
||||
debugInfoSync("✅ MomentListHomeViewModel: 第\(attempt)次重试成功,认证信息已保存,开始获取动态")
|
||||
await performAPICall(isRefresh: isRefresh)
|
||||
return
|
||||
} else {
|
||||
debugInfoSync("⏳ MomentListHomeViewModel: 第\(attempt)次重试,认证信息仍未准备好")
|
||||
}
|
||||
}
|
||||
debugInfoSync("❌ MomentListHomeViewModel: 多次重试后认证信息仍未准备好")
|
||||
await MainActor.run {
|
||||
if isRefresh {
|
||||
self.isLoading = false
|
||||
} else {
|
||||
self.isLoadingMore = false
|
||||
}
|
||||
self.error = "认证信息未准备好"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func performAPICall(isRefresh: Bool) async {
|
||||
let apiService = LiveAPIService()
|
||||
|
||||
do {
|
||||
// 如果是刷新,使用空字符串;如果是加载更多,使用nextDynamicId
|
||||
let dynamicId = isRefresh ? "" : nextDynamicId.description
|
||||
let request = LatestDynamicsRequest(dynamicId: dynamicId, pageSize: 20, types: [.text, .picture])
|
||||
debugInfoSync("📡 MomentListHomeViewModel: 发送请求: \(request.endpoint)")
|
||||
debugInfoSync(" 参数: dynamicId=\(request.dynamicId), pageSize=\(request.pageSize), isRefresh=\(isRefresh)")
|
||||
|
||||
let response: MomentsLatestResponse = try await apiService.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
self.handleAPISuccess(response, isRefresh: isRefresh)
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.handleAPIError(error, isRefresh: isRefresh)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func handleAPISuccess(_ response: MomentsLatestResponse, isRefresh: Bool) {
|
||||
if isRefresh {
|
||||
isLoading = false
|
||||
isLoaded = true
|
||||
} else {
|
||||
isLoadingMore = false
|
||||
}
|
||||
|
||||
debugInfoSync("✅ MomentListHomeViewModel: API 请求成功")
|
||||
debugInfoSync(" 响应码: \(response.code)")
|
||||
debugInfoSync(" 消息: \(response.message)")
|
||||
debugInfoSync(" 数据数量: \(response.data?.dynamicList.count ?? 0)")
|
||||
|
||||
if let list = response.data?.dynamicList {
|
||||
if isRefresh {
|
||||
// 刷新时替换所有数据
|
||||
moments = list
|
||||
debugInfoSync("✅ MomentListHomeViewModel: 数据刷新成功")
|
||||
debugInfoSync(" 动态数量: \(list.count)")
|
||||
} else {
|
||||
// 加载更多时追加数据
|
||||
moments.append(contentsOf: list)
|
||||
debugInfoSync("✅ MomentListHomeViewModel: 数据加载更多成功")
|
||||
debugInfoSync(" 新增动态数量: \(list.count)")
|
||||
debugInfoSync(" 总动态数量: \(moments.count)")
|
||||
}
|
||||
|
||||
// 更新分页信息
|
||||
nextDynamicId = response.data?.nextDynamicId ?? 0
|
||||
hasMore = list.count == 20 // 如果返回的数据少于20条,说明没有更多数据了
|
||||
|
||||
debugInfoSync("📄 MomentListHomeViewModel: 分页信息更新")
|
||||
debugInfoSync(" nextDynamicId: \(nextDynamicId)")
|
||||
debugInfoSync(" hasMore: \(hasMore)")
|
||||
|
||||
error = nil
|
||||
} else {
|
||||
if isRefresh {
|
||||
moments = []
|
||||
}
|
||||
error = response.message
|
||||
debugErrorSync("❌ MomentListHomeViewModel: 数据为空")
|
||||
debugErrorSync(" 错误消息: \(response.message)")
|
||||
}
|
||||
}
|
||||
|
||||
private func handleAPIError(_ error: Error, isRefresh: Bool) {
|
||||
if isRefresh {
|
||||
isLoading = false
|
||||
moments = []
|
||||
} else {
|
||||
isLoadingMore = false
|
||||
}
|
||||
self.error = error.localizedDescription
|
||||
debugErrorSync("❌ MomentListHomeViewModel: API 请求失败")
|
||||
debugErrorSync(" 错误: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
268
yana/MVVM/ViewModel/SettingViewModel.swift
Normal file
268
yana/MVVM/ViewModel/SettingViewModel.swift
Normal file
@@ -0,0 +1,268 @@
|
||||
import SwiftUI
|
||||
import PhotosUI
|
||||
import UIKit
|
||||
|
||||
// MARK: - Setting ViewModel
|
||||
|
||||
@MainActor
|
||||
class SettingViewModel: ObservableObject {
|
||||
// MARK: - Published Properties
|
||||
@Published var userInfo: UserInfo?
|
||||
@Published var isLoadingUserInfo: Bool = false
|
||||
@Published var userInfoError: String?
|
||||
|
||||
// 头像相关
|
||||
@Published var isUploadingAvatar: Bool = false
|
||||
@Published var avatarUploadError: String?
|
||||
|
||||
// 昵称编辑相关
|
||||
@Published var isEditingNickname: Bool = false
|
||||
@Published var nicknameInput: String = ""
|
||||
@Published var isUpdatingUser: Bool = false
|
||||
@Published var updateUserError: String?
|
||||
|
||||
// 图片选择相关
|
||||
@Published var showImageSourceActionSheet: Bool = false
|
||||
@Published var showCamera: Bool = false
|
||||
@Published var showPhotoPicker: Bool = false
|
||||
@Published var selectedPhotoItems: [PhotosPickerItem] = []
|
||||
|
||||
// 弹窗状态
|
||||
@Published var showLogoutConfirmation: Bool = false
|
||||
@Published var showAboutUs: Bool = false
|
||||
@Published var showPrivacyPolicy: Bool = false
|
||||
@Published var showUserAgreement: Bool = false
|
||||
@Published var showDeactivateAccount: Bool = false
|
||||
|
||||
// MARK: - Callbacks
|
||||
var onBack: (() -> Void)?
|
||||
var onLogout: (() -> Void)?
|
||||
|
||||
// MARK: - Private Properties
|
||||
private let apiService: APIServiceProtocol
|
||||
|
||||
// MARK: - Initialization
|
||||
init(apiService: APIServiceProtocol = LiveAPIService()) {
|
||||
self.apiService = apiService
|
||||
}
|
||||
|
||||
// MARK: - Public Methods
|
||||
func onAppear() {
|
||||
debugInfoSync("⚙️ SettingPage onAppear")
|
||||
loadUserInfo()
|
||||
}
|
||||
|
||||
func onBackTapped() {
|
||||
onBack?()
|
||||
}
|
||||
|
||||
// MARK: - User Info Management
|
||||
private func loadUserInfo() {
|
||||
isLoadingUserInfo = true
|
||||
userInfoError = nil
|
||||
|
||||
Task {
|
||||
if let userInfo = await UserInfoManager.getUserInfo() {
|
||||
self.userInfo = userInfo
|
||||
debugInfoSync("✅ 用户信息加载成功")
|
||||
} else {
|
||||
// 尝试从服务器获取
|
||||
if let userInfo = await UserInfoManager.fetchUserInfoFromServer(apiService: apiService) {
|
||||
self.userInfo = userInfo
|
||||
debugInfoSync("✅ 从服务器获取用户信息成功")
|
||||
} else {
|
||||
self.userInfoError = "获取用户信息失败"
|
||||
debugErrorSync("❌ 获取用户信息失败")
|
||||
}
|
||||
}
|
||||
self.isLoadingUserInfo = false
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Avatar Management
|
||||
func onAvatarTapped() {
|
||||
showImageSourceActionSheet = true
|
||||
}
|
||||
|
||||
func selectImageSource(_ source: AppImageSource) {
|
||||
showImageSourceActionSheet = false
|
||||
|
||||
switch source {
|
||||
case .camera:
|
||||
showCamera = true
|
||||
case .photoLibrary:
|
||||
showPhotoPicker = true
|
||||
}
|
||||
}
|
||||
|
||||
func onCameraImagePicked(_ image: UIImage) {
|
||||
showCamera = false
|
||||
uploadAvatar(image)
|
||||
}
|
||||
|
||||
func onPhotoPickerItemsChanged(_ items: [PhotosPickerItem]) {
|
||||
selectedPhotoItems = items
|
||||
|
||||
Task {
|
||||
if let item = items.first {
|
||||
if let data = try? await item.loadTransferable(type: Data.self),
|
||||
let image = UIImage(data: data) {
|
||||
await MainActor.run {
|
||||
showPhotoPicker = false
|
||||
uploadAvatar(image)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func uploadAvatar(_ image: UIImage) {
|
||||
isUploadingAvatar = true
|
||||
avatarUploadError = nil
|
||||
|
||||
Task {
|
||||
if let url = await COSManagerAdapter.shared.uploadUIImage(image, apiService: apiService) {
|
||||
await MainActor.run {
|
||||
self.isUploadingAvatar = false
|
||||
self.updateUserAvatar(url)
|
||||
}
|
||||
} else {
|
||||
await MainActor.run {
|
||||
self.isUploadingAvatar = false
|
||||
self.avatarUploadError = "头像上传失败"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private func updateUserAvatar(_ avatarUrl: String) {
|
||||
guard let userInfo = userInfo else { return }
|
||||
|
||||
isUpdatingUser = true
|
||||
updateUserError = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let ticket = await UserInfoManager.getCurrentUserTicket() ?? ""
|
||||
let request = UpdateUserRequest(avatar: avatarUrl, nick: nil, uid: userInfo.uid ?? 0, ticket: ticket)
|
||||
let response: UpdateUserResponse = try await apiService.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
self.isUpdatingUser = false
|
||||
if response.code == 200 {
|
||||
// 刷新用户信息
|
||||
self.loadUserInfo()
|
||||
} else {
|
||||
self.updateUserError = response.message
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.isUpdatingUser = false
|
||||
self.updateUserError = error.localizedDescription
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Nickname Management
|
||||
func onNicknameTapped() {
|
||||
nicknameInput = userInfo?.nick ?? ""
|
||||
isEditingNickname = true
|
||||
}
|
||||
|
||||
func onNicknameInputChanged(_ text: String) {
|
||||
nicknameInput = String(text.prefix(15))
|
||||
}
|
||||
|
||||
func onNicknameEditConfirmed() {
|
||||
let trimmed = nicknameInput.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
guard !trimmed.isEmpty else { return }
|
||||
|
||||
isEditingNickname = false
|
||||
updateUserNickname(trimmed)
|
||||
}
|
||||
|
||||
private func updateUserNickname(_ nickname: String) {
|
||||
guard let userInfo = userInfo else { return }
|
||||
|
||||
isUpdatingUser = true
|
||||
updateUserError = nil
|
||||
|
||||
Task {
|
||||
do {
|
||||
let ticket = await UserInfoManager.getCurrentUserTicket() ?? ""
|
||||
let request = UpdateUserRequest(avatar: nil, nick: nickname, uid: userInfo.uid ?? 0, ticket: ticket)
|
||||
let response: UpdateUserResponse = try await apiService.request(request)
|
||||
|
||||
await MainActor.run {
|
||||
self.isUpdatingUser = false
|
||||
if response.code == 200 {
|
||||
// 刷新用户信息
|
||||
self.loadUserInfo()
|
||||
} else {
|
||||
self.updateUserError = response.message
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
await MainActor.run {
|
||||
self.isUpdatingUser = false
|
||||
self.updateUserError = error.localizedDescription
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Settings Actions
|
||||
func onPersonalInfoPermissionsTapped() {
|
||||
showPrivacyPolicy = true
|
||||
}
|
||||
|
||||
func onHelpTapped() {
|
||||
showUserAgreement = true
|
||||
}
|
||||
|
||||
func onClearCacheTapped() {
|
||||
// TODO: 实现清除缓存逻辑
|
||||
debugInfoSync("🗑️ 清除缓存")
|
||||
}
|
||||
|
||||
func onCheckUpdatesTapped() {
|
||||
// TODO: 实现检查更新逻辑
|
||||
debugInfoSync("🔄 检查更新")
|
||||
}
|
||||
|
||||
func onDeactivateAccountTapped() {
|
||||
showDeactivateAccount = true
|
||||
}
|
||||
|
||||
func onAboutUsTapped() {
|
||||
showAboutUs = true
|
||||
}
|
||||
|
||||
func onLogoutTapped() {
|
||||
showLogoutConfirmation = true
|
||||
}
|
||||
|
||||
func onLogoutConfirmed() {
|
||||
Task {
|
||||
await UserInfoManager.clearAllAuthenticationData()
|
||||
await MainActor.run {
|
||||
onLogout?()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - WebView Dismissal
|
||||
func onPrivacyPolicyDismissed() {
|
||||
showPrivacyPolicy = false
|
||||
}
|
||||
|
||||
func onUserAgreementDismissed() {
|
||||
showUserAgreement = false
|
||||
}
|
||||
|
||||
func onDeactivateAccountDismissed() {
|
||||
showDeactivateAccount = false
|
||||
}
|
||||
}
|
||||
@@ -12,6 +12,9 @@
|
||||
"login.agreement_policy" = "Agree to the \"User Service Agreement\" and \"Privacy Policy\"";
|
||||
"login.agreement" = "User Service Agreement";
|
||||
"login.policy" = "Privacy Policy";
|
||||
"login.agreement_alert_title" = "Notice";
|
||||
"login.agreement_alert_message" = "Please agree to the User Service Agreement and Privacy Policy first";
|
||||
"login.agreement_alert_confirm" = "OK";
|
||||
|
||||
// MARK: - Common Buttons
|
||||
"common.login" = "Login";
|
||||
@@ -37,6 +40,9 @@
|
||||
"id_login.forgot_password" = "Forgot Password?";
|
||||
"id_login.login_button" = "Login";
|
||||
"id_login.logging_in" = "Logging in...";
|
||||
"id_login.password" = "Password";
|
||||
"id_login.login" = "Login";
|
||||
"id_login.user_id" = "User ID";
|
||||
|
||||
// MARK: - Email Login Page
|
||||
"email_login.title" = "Email Login";
|
||||
@@ -48,6 +54,9 @@
|
||||
"email_login.code_sent" = "Verification code sent";
|
||||
"email_login.login_button" = "Login";
|
||||
"email_login.logging_in" = "Logging in...";
|
||||
"email_login.email" = "Email";
|
||||
"email_login.verification_code" = "Verification Code";
|
||||
"email_login.login" = "Login";
|
||||
"placeholder.enter_email" = "Please enter email";
|
||||
"placeholder.enter_verification_code" = "Please enter verification code";
|
||||
|
||||
@@ -81,16 +90,17 @@
|
||||
"createFeed.processingImages" = "Processing images...";
|
||||
"createFeed.publishing" = "Publishing...";
|
||||
"createFeed.publish" = "Publish";
|
||||
"createFeed.title" = "Image & Text Publish";
|
||||
"createFeed.title" = "Image & Text";
|
||||
|
||||
// MARK: - Edit Feed
|
||||
"editFeed.title" = "Image & Text Edit";
|
||||
"editFeed.title" = "Image & Text";
|
||||
"editFeed.publish" = "Publish";
|
||||
"editFeed.enterContent" = "Enter Content";
|
||||
|
||||
// MARK: - Feed List
|
||||
"feedList.title" = "Enjoy your Life Time";
|
||||
"feedList.slogan" = "The disease is like a cruel ruler,\nand time is our most precious treasure.\nEvery moment we live is a victory\nagainst the inevitable.";
|
||||
"feedList.empty" = "No moments yet";
|
||||
|
||||
// MARK: - Feed
|
||||
"feed.title" = "Enjoy your Life Time";
|
||||
@@ -129,4 +139,90 @@
|
||||
"appSetting.checkUpdates" = "Check for Updates";
|
||||
"appSetting.logout" = "Log Out";
|
||||
"appSetting.aboutUs" = "About Us";
|
||||
"appSetting.logoutAccount" = "Log out of account";
|
||||
"appSetting.aboutUs.title" = "About Us";
|
||||
"appSetting.logoutConfirmation.title" = "Confirm Logout";
|
||||
"appSetting.logoutConfirmation.confirm" = "Confirm Logout";
|
||||
"appSetting.logoutConfirmation.message" = "Are you sure you want to logout from your current account?";
|
||||
"appSetting.deactivateAccount" = "Deactivate Account";
|
||||
"appSetting.logoutAccount" = "Log out of account";
|
||||
"app_settings.not_set" = "Not set";
|
||||
|
||||
// MARK: - Detail
|
||||
"detail.title" = "Enjoy your life";
|
||||
|
||||
// MARK: - Edit Feed
|
||||
"edit_feed.uploading_progress" = "Uploading images...%d%%";
|
||||
|
||||
// MARK: - Web View
|
||||
"web_view.load_failed" = "Failed to load page";
|
||||
"web_view.open_webpage" = "Open Webpage";
|
||||
|
||||
// MARK: - Language Settings
|
||||
"language_settings.select_language" = "Select Language";
|
||||
"language_settings.current_language" = "Current Language";
|
||||
"language_settings.language_info" = "Language Info";
|
||||
"language_settings.test_area" = "Language Switch Test";
|
||||
"language_settings.test_region" = "Test Area";
|
||||
"language_settings.token_success" = "✅ Token obtained successfully";
|
||||
"language_settings.bucket" = "Bucket: %@";
|
||||
"language_settings.region" = "Region: %@";
|
||||
"language_settings.app_id" = "App ID: %@";
|
||||
"language_settings.custom_domain" = "Custom Domain: %@";
|
||||
"language_settings.accelerate_enabled" = "Enabled";
|
||||
"language_settings.accelerate_disabled" = "Disabled";
|
||||
"language_settings.accelerate_status" = "Acceleration: %@";
|
||||
"language_settings.expiration_date" = "Expiration Date: %@";
|
||||
"language_settings.remaining_time" = "Remaining Time: %d seconds";
|
||||
"language_settings.test_cos_token" = "Test Tencent Cloud COS Token";
|
||||
"language_settings.title" = "Language Settings";
|
||||
|
||||
// MARK: - App Settings
|
||||
"app_settings.error" = "Error";
|
||||
"app_settings.confirm" = "Confirm";
|
||||
"app_settings.nickname_limit" = "Nickname must be 15 characters or less";
|
||||
"app_settings.take_photo" = "Take Photo";
|
||||
"app_settings.select_from_album" = "Select from Album";
|
||||
|
||||
// MARK: - Test
|
||||
"test.test_page" = "Test Page";
|
||||
"test.test_description" = "This is a test page\nfor verifying navigation functionality";
|
||||
"test.test_button" = "Test Button";
|
||||
"test.back" = "Back";
|
||||
|
||||
// MARK: - Image Picker
|
||||
"image_picker.loading_image" = "Loading image...";
|
||||
"image_picker.cancel" = "Cancel";
|
||||
"image_picker.confirm" = "Confirm";
|
||||
|
||||
// MARK: - Content View
|
||||
"content_view.log_level" = "Log Level:";
|
||||
"content_view.no_log" = "No Log";
|
||||
"content_view.basic_log" = "Basic Log";
|
||||
"content_view.detailed_log" = "Detailed Log";
|
||||
"content_view.api_test_result" = "API Test Result:";
|
||||
"content_view.status" = "Status: %@";
|
||||
"content_view.message" = "Message: %@";
|
||||
"content_view.version" = "Version: %@";
|
||||
"content_view.unknown" = "Unknown";
|
||||
"content_view.timestamp" = "Timestamp: %d";
|
||||
"content_view.config" = "Configuration:";
|
||||
|
||||
// MARK: - Screen Adapter
|
||||
"screen_adapter.method1" = "Method 1: Direct Call";
|
||||
"screen_adapter.method2" = "Method 2: View Extension";
|
||||
"screen_adapter.method3" = "Method 3: Ratio Calculation";
|
||||
|
||||
// MARK: - Config
|
||||
"config.api_test" = "API Configuration Test";
|
||||
"config.loading" = "Loading configuration...";
|
||||
"config.error" = "Error";
|
||||
"config.feature_list" = "Feature List";
|
||||
"config.settings" = "Settings";
|
||||
"config.last_updated" = "Last Updated: %@";
|
||||
"config.click_to_load" = "Click the button below to load configuration";
|
||||
"config.use_new_tca" = "Use new TCA API component";
|
||||
"config.clear_error" = "Clear Error";
|
||||
"config.version" = "Version";
|
||||
"config.debug_mode" = "Debug Mode";
|
||||
"config.api_timeout" = "API Timeout";
|
||||
"config.max_retries" = "Max Retries";
|
||||
|
||||
@@ -13,6 +13,9 @@
|
||||
"login.agreement_policy" = "同意《用戶服務協議》和《隱私政策》";
|
||||
"login.agreement" = "《用戶服務協議》";
|
||||
"login.policy" = "《隱私政策》";
|
||||
"login.agreement_alert_title" = "提示";
|
||||
"login.agreement_alert_message" = "请先同意用户服务协议和隐私政策";
|
||||
"login.agreement_alert_confirm" = "确定";
|
||||
|
||||
// MARK: - 通用按钮
|
||||
"common.login" = "登录";
|
||||
@@ -38,6 +41,9 @@
|
||||
"id_login.forgot_password" = "忘记密码?";
|
||||
"id_login.login_button" = "登录";
|
||||
"id_login.logging_in" = "登录中...";
|
||||
"id_login.password" = "密码";
|
||||
"id_login.login" = "登录";
|
||||
"id_login.user_id" = "用户ID";
|
||||
|
||||
// MARK: - 邮箱登录页面
|
||||
"email_login.title" = "邮箱登录";
|
||||
@@ -49,6 +55,9 @@
|
||||
"email_login.code_sent" = "验证码已发送";
|
||||
"email_login.login_button" = "登录";
|
||||
"email_login.logging_in" = "登录中...";
|
||||
"email_login.email" = "邮箱";
|
||||
"email_login.verification_code" = "验证码";
|
||||
"email_login.login" = "登录";
|
||||
"placeholder.enter_email" = "请输入邮箱";
|
||||
"placeholder.enter_verification_code" = "请输入验证码";
|
||||
|
||||
@@ -88,7 +97,8 @@
|
||||
"editFeed.enterContent" = "输入内容";
|
||||
|
||||
"feedList.title" = "享受您的生活时光";
|
||||
"feedList.slogan" = "疾病如同残酷的统治者,\n而时间是我们最宝贵的财富。\n我们活着的每一刻,都是对不可避免命运的胜利。";
|
||||
"feedList.slogan" = "疾病如同残酷的统治者,\n而时间是我们最宝贵的财富。\n我们活着的每一刻,都是对不可避免命运的胜利。";
|
||||
"feedList.empty" = "暂无动态";
|
||||
|
||||
"feed.title" = "享受您的生活时光";
|
||||
"feed.empty" = "暂无动态内容";
|
||||
@@ -125,4 +135,90 @@
|
||||
"appSetting.checkUpdates" = "检查更新";
|
||||
"appSetting.logout" = "退出登录";
|
||||
"appSetting.aboutUs" = "关于我们";
|
||||
"appSetting.logoutAccount" = "退出账户";
|
||||
"appSetting.aboutUs.title" = "关于我们";
|
||||
"appSetting.logoutConfirmation.title" = "确认退出";
|
||||
"appSetting.logoutConfirmation.confirm" = "确认退出";
|
||||
"appSetting.logoutConfirmation.message" = "确定要退出当前账户吗?";
|
||||
"appSetting.deactivateAccount" = "注销帐号";
|
||||
"appSetting.logoutAccount" = "退出账户";
|
||||
"app_settings.not_set" = "未设置";
|
||||
|
||||
// MARK: - Detail
|
||||
"detail.title" = "享受你的生活";
|
||||
|
||||
// MARK: - Edit Feed
|
||||
"edit_feed.uploading_progress" = "正在上传图片...%d%%";
|
||||
|
||||
// MARK: - Web View
|
||||
"web_view.load_failed" = "无法加载页面";
|
||||
"web_view.open_webpage" = "打开网页";
|
||||
|
||||
// MARK: - Language Settings
|
||||
"language_settings.select_language" = "选择语言";
|
||||
"language_settings.current_language" = "当前语言";
|
||||
"language_settings.language_info" = "语言信息";
|
||||
"language_settings.test_area" = "语言切换测试";
|
||||
"language_settings.test_region" = "测试区域";
|
||||
"language_settings.token_success" = "✅ Token 获取成功";
|
||||
"language_settings.bucket" = "存储桶: %@";
|
||||
"language_settings.region" = "地域: %@";
|
||||
"language_settings.app_id" = "应用ID: %@";
|
||||
"language_settings.custom_domain" = "自定义域名: %@";
|
||||
"language_settings.accelerate_enabled" = "启用";
|
||||
"language_settings.accelerate_disabled" = "禁用";
|
||||
"language_settings.accelerate_status" = "加速: %@";
|
||||
"language_settings.expiration_date" = "过期时间: %@";
|
||||
"language_settings.remaining_time" = "剩余时间: %d秒";
|
||||
"language_settings.test_cos_token" = "测试腾讯云 COS Token";
|
||||
"language_settings.title" = "语言设置";
|
||||
|
||||
// MARK: - App Settings
|
||||
"app_settings.error" = "错误";
|
||||
"app_settings.confirm" = "确定";
|
||||
"app_settings.nickname_limit" = "昵称最长15个字符";
|
||||
"app_settings.take_photo" = "拍照";
|
||||
"app_settings.select_from_album" = "从相册选择";
|
||||
|
||||
// MARK: - Test
|
||||
"test.test_page" = "测试页面";
|
||||
"test.test_description" = "这是一个测试用的页面\n用于验证导航跳转功能";
|
||||
"test.test_button" = "测试按钮";
|
||||
"test.back" = "返回";
|
||||
|
||||
// MARK: - Image Picker
|
||||
"image_picker.loading_image" = "加载图片中...";
|
||||
"image_picker.cancel" = "取消";
|
||||
"image_picker.confirm" = "确认";
|
||||
|
||||
// MARK: - Content View
|
||||
"content_view.log_level" = "日志级别:";
|
||||
"content_view.no_log" = "无日志";
|
||||
"content_view.basic_log" = "基础日志";
|
||||
"content_view.detailed_log" = "详细日志";
|
||||
"content_view.api_test_result" = "API 测试结果:";
|
||||
"content_view.status" = "状态: %@";
|
||||
"content_view.message" = "消息: %@";
|
||||
"content_view.version" = "版本: %@";
|
||||
"content_view.unknown" = "未知";
|
||||
"content_view.timestamp" = "时间戳: %d";
|
||||
"content_view.config" = "配置:";
|
||||
|
||||
// MARK: - Screen Adapter
|
||||
"screen_adapter.method1" = "方法1: 直接调用";
|
||||
"screen_adapter.method2" = "方法2: View Extension";
|
||||
"screen_adapter.method3" = "方法3: 比例计算";
|
||||
|
||||
// MARK: - Config
|
||||
"config.api_test" = "API 配置测试";
|
||||
"config.loading" = "正在加载配置...";
|
||||
"config.error" = "错误";
|
||||
"config.feature_list" = "功能列表";
|
||||
"config.settings" = "设置";
|
||||
"config.last_updated" = "最后更新: %@";
|
||||
"config.click_to_load" = "点击下方按钮加载配置";
|
||||
"config.use_new_tca" = "使用新的 TCA API 组件";
|
||||
"config.clear_error" = "清除错误";
|
||||
"config.version" = "版本";
|
||||
"config.debug_mode" = "调试模式";
|
||||
"config.api_timeout" = "API 超时";
|
||||
"config.max_retries" = "最大重试次数";
|
||||
|
||||
@@ -3,93 +3,62 @@ import SwiftUI
|
||||
// MARK: - API Loading Effect View
|
||||
|
||||
/// 全局 API 加载效果视图
|
||||
///
|
||||
/// 该视图显示在屏幕最顶层,包含:
|
||||
/// - Loading 动画(88x88,60% alpha 黑色圆角背景)
|
||||
/// - 错误信息显示(2秒后自动消失)
|
||||
/// - 支持多个并发显示
|
||||
/// - 不阻挡用户点击操作
|
||||
struct APILoadingEffectView: View {
|
||||
@ObservedObject private var loadingManager = APILoadingManager.shared
|
||||
|
||||
var body: some View {
|
||||
ZStack {
|
||||
// 🚨 极简渲染策略:避免复杂的 ForEach,只显示第一个需要显示的项目
|
||||
if let firstItem = getFirstDisplayItem() {
|
||||
SingleLoadingView(item: firstItem)
|
||||
.onAppear {
|
||||
debugInfoSync("🔍 Loading item appeared: \(firstItem.id)")
|
||||
}
|
||||
.onDisappear {
|
||||
debugInfoSync("🔍 Loading item disappeared: \(firstItem.id)")
|
||||
}
|
||||
LoadingItemView(item: firstItem)
|
||||
}
|
||||
}
|
||||
.allowsHitTesting(false) // 不阻挡用户点击
|
||||
.ignoresSafeArea(.all) // 覆盖整个屏幕
|
||||
.onReceive(loadingManager.$loadingItems) { items in
|
||||
debugInfoSync("🔍 Loading items updated: \(items.count) items")
|
||||
}
|
||||
.allowsHitTesting(false)
|
||||
.ignoresSafeArea(.all)
|
||||
}
|
||||
|
||||
/// 安全地获取第一个需要显示的项目
|
||||
private func getFirstDisplayItem() -> APILoadingItem? {
|
||||
guard Thread.isMainThread else {
|
||||
debugWarnSync("⚠️ getFirstDisplayItem called from background thread")
|
||||
return nil
|
||||
}
|
||||
|
||||
guard Thread.isMainThread else { return nil }
|
||||
return loadingManager.loadingItems.first { $0.shouldDisplay }
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Single Loading View
|
||||
// MARK: - Loading Item View
|
||||
|
||||
/// 单个加载项视图 - 极简版本
|
||||
private struct SingleLoadingView: View {
|
||||
private struct LoadingItemView: View {
|
||||
let item: APILoadingItem
|
||||
|
||||
var body: some View {
|
||||
Group {
|
||||
switch item.state {
|
||||
case .loading:
|
||||
SimpleLoadingView()
|
||||
|
||||
case .error(let message):
|
||||
if item.shouldShowError {
|
||||
SimpleErrorView(message: message)
|
||||
}
|
||||
|
||||
case .success:
|
||||
EmptyView() // 成功状态不显示任何内容
|
||||
switch item.state {
|
||||
case .loading:
|
||||
LoadingSpinnerView()
|
||||
case .error(let message):
|
||||
if item.shouldShowError {
|
||||
ErrorMessageView(message: message)
|
||||
} else {
|
||||
EmptyView()
|
||||
}
|
||||
case .success:
|
||||
EmptyView()
|
||||
}
|
||||
// 🚨 移除复杂动画,避免渲染问题
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Simple Loading View
|
||||
// MARK: - Loading Spinner View
|
||||
|
||||
/// 极简 Loading 视图
|
||||
private struct SimpleLoadingView: View {
|
||||
private struct LoadingSpinnerView: View {
|
||||
var body: some View {
|
||||
VStack {
|
||||
Spacer()
|
||||
HStack {
|
||||
Spacer()
|
||||
|
||||
// 极简黑色背景 + 白色圆圈
|
||||
ZStack {
|
||||
RoundedRectangle(cornerRadius: 12)
|
||||
.fill(Color.black.opacity(0.6))
|
||||
.frame(width: 88, height: 88)
|
||||
|
||||
// 使用最简单的 ProgressView
|
||||
ProgressView()
|
||||
.progressViewStyle(CircularProgressViewStyle(tint: .white))
|
||||
.scaleEffect(1.2)
|
||||
}
|
||||
|
||||
Spacer()
|
||||
}
|
||||
Spacer()
|
||||
@@ -97,10 +66,9 @@ private struct SimpleLoadingView: View {
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Simple Error View
|
||||
// MARK: - Error Message View
|
||||
|
||||
/// 极简错误视图
|
||||
private struct SimpleErrorView: View {
|
||||
private struct ErrorMessageView: View {
|
||||
let message: String
|
||||
|
||||
var body: some View {
|
||||
@@ -108,13 +76,10 @@ private struct SimpleErrorView: View {
|
||||
Spacer()
|
||||
HStack {
|
||||
Spacer()
|
||||
|
||||
// 极简错误提示
|
||||
VStack(spacing: 8) {
|
||||
Image(systemName: "exclamationmark.triangle.fill")
|
||||
.foregroundColor(.white)
|
||||
.font(.title2)
|
||||
|
||||
Text(message)
|
||||
.foregroundColor(.white)
|
||||
.font(.system(size: 14))
|
||||
@@ -127,101 +92,9 @@ private struct SimpleErrorView: View {
|
||||
.fill(Color.black.opacity(0.6))
|
||||
)
|
||||
.frame(maxWidth: 250)
|
||||
|
||||
Spacer()
|
||||
}
|
||||
Spacer()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Preview
|
||||
|
||||
#if DEBUG
|
||||
struct APILoadingEffectView_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
ZStack {
|
||||
// 模拟背景
|
||||
Rectangle()
|
||||
.fill(Color.blue.opacity(0.3))
|
||||
.ignoresSafeArea()
|
||||
|
||||
VStack(spacing: 20) {
|
||||
Text("背景内容")
|
||||
.font(.title)
|
||||
|
||||
Button("测试按钮") {
|
||||
debugInfoSync("按钮被点击了!")
|
||||
}
|
||||
.padding()
|
||||
.background(Color.blue)
|
||||
.foregroundColor(.white)
|
||||
.cornerRadius(8)
|
||||
}
|
||||
|
||||
// Loading Effect View
|
||||
APILoadingEffectView()
|
||||
}
|
||||
.previewDisplayName("API Loading Effect")
|
||||
.onAppear {
|
||||
// 模拟不同状态的预览
|
||||
Task {
|
||||
let manager = APILoadingManager.shared
|
||||
|
||||
// 添加 loading
|
||||
let id1 = manager.startLoading()
|
||||
|
||||
// 2秒后添加错误
|
||||
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
|
||||
Task {
|
||||
manager.setError(id1, errorMessage: "网络连接失败,请检查网络设置")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Preview Helpers
|
||||
|
||||
/// 预览用的测试状态
|
||||
private struct PreviewStateModifier: ViewModifier {
|
||||
let showLoading: Bool
|
||||
let showError: Bool
|
||||
let errorMessage: String
|
||||
|
||||
func body(content: Content) -> some View {
|
||||
content
|
||||
.onAppear {
|
||||
Task {
|
||||
let manager = APILoadingManager.shared
|
||||
|
||||
if showLoading {
|
||||
let _ = manager.startLoading()
|
||||
}
|
||||
|
||||
if showError {
|
||||
let id = manager.startLoading()
|
||||
try? await Task.sleep(nanoseconds: 1_000_000_000) // 1秒
|
||||
manager.setError(id, errorMessage: errorMessage)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
extension View {
|
||||
/// 添加预览状态
|
||||
func previewLoadingState(
|
||||
showLoading: Bool = false,
|
||||
showError: Bool = false,
|
||||
errorMessage: String = "示例错误信息"
|
||||
) -> some View {
|
||||
self.modifier(PreviewStateModifier(
|
||||
showLoading: showLoading,
|
||||
showError: showError,
|
||||
errorMessage: errorMessage
|
||||
))
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
@@ -134,4 +134,18 @@ extension APILoadingManager {
|
||||
return .failure(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Global Convenience Methods
|
||||
|
||||
/// 全局便捷方法:同步设置错误信息(fire-and-forget)
|
||||
/// - Parameters:
|
||||
/// - id: 加载 ID
|
||||
/// - errorMessage: 错误信息
|
||||
func setAPILoadingErrorSync(_ id: UUID, errorMessage: String) {
|
||||
Task {
|
||||
await MainActor.run {
|
||||
APILoadingManager.shared.setError(id, errorMessage: errorMessage)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,241 +0,0 @@
|
||||
import Foundation
|
||||
import QCloudCOSXML
|
||||
|
||||
// MARK: - 腾讯云 COS 管理器
|
||||
|
||||
/// 腾讯云 COS 管理器
|
||||
///
|
||||
/// 负责管理腾讯云 COS 相关的操作,包括:
|
||||
/// - Token 获取和缓存
|
||||
/// - 文件上传、下载、删除
|
||||
/// - 凭证管理和过期处理
|
||||
@MainActor
|
||||
class COSManager: ObservableObject {
|
||||
static let shared = COSManager()
|
||||
|
||||
private init() {}
|
||||
|
||||
// 幂等初始化标记
|
||||
private static var isCOSInitialized = false
|
||||
|
||||
// 幂等初始化方法
|
||||
private func ensureCOSInitialized(tokenData: TcTokenData) {
|
||||
guard !Self.isCOSInitialized else { return }
|
||||
let configuration = QCloudServiceConfiguration()
|
||||
let endpoint = QCloudCOSXMLEndPoint()
|
||||
endpoint.regionName = tokenData.region
|
||||
endpoint.useHTTPS = true
|
||||
if tokenData.accelerate {
|
||||
endpoint.suffix = "cos.accelerate.myqcloud.com"
|
||||
}
|
||||
configuration.endpoint = endpoint
|
||||
QCloudCOSXMLService.registerDefaultCOSXML(with: configuration)
|
||||
QCloudCOSTransferMangerService.registerDefaultCOSTransferManger(with: configuration)
|
||||
Self.isCOSInitialized = true
|
||||
debugInfoSync("✅ COS服务已初始化,region: \(tokenData.region)")
|
||||
}
|
||||
|
||||
// MARK: - Token 管理
|
||||
|
||||
/// 当前缓存的 Token 信息
|
||||
private var cachedToken: TcTokenData?
|
||||
private var tokenExpirationDate: Date?
|
||||
|
||||
/// 获取腾讯云 COS Token
|
||||
/// - Parameter apiService: API 服务实例
|
||||
/// - Returns: Token 数据,如果获取失败返回 nil
|
||||
func getToken(apiService: any APIServiceProtocol & Sendable) async -> TcTokenData? {
|
||||
// 检查缓存是否有效
|
||||
if let cached = cachedToken, let expiration = tokenExpirationDate, Date() < expiration {
|
||||
debugInfoSync("🔐 使用缓存的 COS Token")
|
||||
return cached
|
||||
}
|
||||
|
||||
// 清除过期缓存
|
||||
clearCachedToken()
|
||||
|
||||
// 请求新的 Token
|
||||
debugInfoSync("🔐 开始请求腾讯云 COS Token...")
|
||||
|
||||
do {
|
||||
let request = TcTokenRequest()
|
||||
let response: TcTokenResponse = try await apiService.request(request)
|
||||
|
||||
guard response.code == 200, let tokenData = response.data else {
|
||||
debugInfoSync("❌ COS Token 请求失败: \(response.message)")
|
||||
return nil
|
||||
}
|
||||
|
||||
// 缓存 Token 和过期时间
|
||||
cachedToken = tokenData
|
||||
tokenExpirationDate = tokenData.expirationDate
|
||||
|
||||
debugInfoSync("✅ COS Token 获取成功")
|
||||
debugInfoSync(" - 存储桶: \(tokenData.bucket)")
|
||||
debugInfoSync(" - 地域: \(tokenData.region)")
|
||||
debugInfoSync(" - 过期时间: \(tokenData.expirationDate)")
|
||||
debugInfoSync(" - 剩余时间: \(tokenData.remainingTime)秒")
|
||||
|
||||
return tokenData
|
||||
|
||||
} catch {
|
||||
debugInfoSync("❌ COS Token 请求异常: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
/// 缓存 Token 信息
|
||||
/// - Parameter tokenData: Token 数据
|
||||
private func cacheToken(_ tokenData: TcTokenData) async {
|
||||
cachedToken = tokenData
|
||||
|
||||
// 解析过期时间(假设 expiration 是 ISO 8601 格式)
|
||||
if let expirationDate = ISO8601DateFormatter().date(from: String(tokenData.expireTime)) {
|
||||
// 提前 5 分钟过期,确保安全
|
||||
tokenExpirationDate = expirationDate.addingTimeInterval(-300)
|
||||
} else {
|
||||
// 如果解析失败,设置默认过期时间(1小时)
|
||||
tokenExpirationDate = Date().addingTimeInterval(3600)
|
||||
}
|
||||
|
||||
debugInfoSync("💾 COS Token 已缓存,过期时间: \(tokenExpirationDate?.description ?? "未知")")
|
||||
}
|
||||
|
||||
/// 清除缓存的 Token
|
||||
private func clearCachedToken() {
|
||||
cachedToken = nil
|
||||
tokenExpirationDate = nil
|
||||
debugInfoSync("🗑️ 清除缓存的 COS Token")
|
||||
}
|
||||
|
||||
/// 强制刷新 Token
|
||||
func refreshToken(apiService: any APIServiceProtocol & Sendable) async -> TcTokenData? {
|
||||
clearCachedToken()
|
||||
return await getToken(apiService: apiService)
|
||||
}
|
||||
|
||||
// MARK: - 只读属性
|
||||
/// 外部安全访问 Token
|
||||
var token: TcTokenData? { cachedToken }
|
||||
|
||||
// MARK: - 调试信息
|
||||
|
||||
/// 获取当前 Token 状态信息
|
||||
func getTokenStatus() -> String {
|
||||
if let _ = cachedToken, let expiration = tokenExpirationDate {
|
||||
let isExpired = Date() >= expiration
|
||||
return "Token 状态: \(isExpired ? "已过期" : "有效"), 过期时间: \(expiration)"
|
||||
} else {
|
||||
return "Token 状态: 未缓存"
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 上传功能
|
||||
|
||||
/// 上传图片到腾讯云 COS
|
||||
/// - Parameters:
|
||||
/// - imageData: 图片数据
|
||||
/// - apiService: API 服务实例
|
||||
/// - Returns: 上传成功的云地址,如果失败返回 nil
|
||||
func uploadImage(_ imageData: Data, apiService: any APIServiceProtocol & Sendable) async -> String? {
|
||||
guard let tokenData = await getToken(apiService: apiService) else {
|
||||
debugInfoSync("❌ 无法获取 COS Token")
|
||||
return nil
|
||||
}
|
||||
// 上传前确保COS服务已初始化
|
||||
ensureCOSInitialized(tokenData: tokenData)
|
||||
|
||||
// 初始化 COS 配置
|
||||
let credential = QCloudCredential()
|
||||
credential.secretID = tokenData.secretId
|
||||
// 打印secretKey原始内容,去除首尾空白
|
||||
let rawSecretKey = tokenData.secretKey.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
debugInfoSync("secretKey原始内容: [\(rawSecretKey)]")
|
||||
credential.secretKey = rawSecretKey
|
||||
credential.token = tokenData.sessionToken
|
||||
credential.startDate = tokenData.startDate
|
||||
credential.expirationDate = tokenData.expirationDate
|
||||
|
||||
let request = QCloudCOSXMLUploadObjectRequest<AnyObject>()
|
||||
request.bucket = tokenData.bucket
|
||||
request.regionName = tokenData.region
|
||||
request.credential = credential
|
||||
|
||||
// 生成唯一 key
|
||||
let fileExtension = "jpg" // 假设为 JPG,可根据实际调整
|
||||
let key = "images/\(UUID().uuidString).\(fileExtension)"
|
||||
request.object = key
|
||||
request.body = imageData as AnyObject
|
||||
|
||||
//监听上传进度
|
||||
request.sendProcessBlock = { (bytesSent, totalBytesSent,
|
||||
totalBytesExpectedToSend) in
|
||||
debugInfoSync("\(bytesSent), \(totalBytesSent), \(totalBytesExpectedToSend)")
|
||||
// bytesSent 本次要发送的字节数(一个大文件可能要分多次发送)
|
||||
// totalBytesSent 已发送的字节数
|
||||
// totalBytesExpectedToSend 本次上传要发送的总字节数(即一个文件大小)
|
||||
};
|
||||
|
||||
// 设置加速
|
||||
if tokenData.accelerate {
|
||||
request.enableQuic = true
|
||||
// endpoint 增加 "cos.accelerate.myqcloud.com"
|
||||
}
|
||||
|
||||
// 使用 async/await 包装上传回调
|
||||
return await withCheckedContinuation { continuation in
|
||||
request.setFinish { result, error in
|
||||
if let error = error {
|
||||
debugInfoSync("❌ 图片上传失败: \(error.localizedDescription)")
|
||||
continuation.resume(returning: " ?????????? ")
|
||||
} else {
|
||||
// 构建云地址
|
||||
let domain = tokenData.customDomain.isEmpty ? "\(tokenData.bucket).cos.\(tokenData.region).myqcloud.com" : tokenData.customDomain
|
||||
let prefix = domain.hasPrefix("http") ? "" : "https://"
|
||||
let cloudURL = "\(prefix)\(domain)/\(key)"
|
||||
debugInfoSync("✅ 图片上传成功: \(cloudURL)")
|
||||
continuation.resume(returning: cloudURL)
|
||||
}
|
||||
}
|
||||
QCloudCOSTransferMangerService.defaultCOSTransferManager().uploadObject(request)
|
||||
}
|
||||
}
|
||||
|
||||
/// 上传 UIImage 到腾讯云 COS,自动压缩为 JPEG(0.8)
|
||||
/// - Parameters:
|
||||
/// - image: UIImage 实例
|
||||
/// - apiService: API 服务实例
|
||||
/// - Returns: 上传成功的云地址,如果失败返回 nil
|
||||
func uploadUIImage(_ image: UIImage, apiService: any APIServiceProtocol & Sendable) async -> String? {
|
||||
guard let data = image.jpegData(compressionQuality: 0.8) else {
|
||||
debugInfoSync("❌ 图片压缩失败,无法生成 JPEG 数据")
|
||||
return nil
|
||||
}
|
||||
return await uploadImage(data, apiService: apiService)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 调试扩展
|
||||
|
||||
extension COSManager {
|
||||
/// 测试 Token 获取功能(仅用于调试)
|
||||
func testTokenRetrieval(apiService: any APIServiceProtocol & Sendable) async {
|
||||
#if DEBUG
|
||||
debugInfoSync("\n🧪 开始测试腾讯云 COS Token 获取功能")
|
||||
|
||||
let token = await getToken(apiService: apiService)
|
||||
if let tokenData = token {
|
||||
debugInfoSync("✅ Token 获取成功")
|
||||
debugInfoSync(" bucket: \(tokenData.bucket)")
|
||||
debugInfoSync(" Expiration: \(tokenData.expireTime)")
|
||||
debugInfoSync(" Token: \(tokenData.sessionToken.prefix(20))...")
|
||||
debugInfoSync(" SecretId: \(tokenData.secretId.prefix(20))...")
|
||||
} else {
|
||||
debugInfoSync("❌ Token 获取失败")
|
||||
}
|
||||
|
||||
debugInfoSync("📊 Token 状态: \(getTokenStatus())")
|
||||
debugInfoSync("✅ 腾讯云 COS Token 测试完成\n")
|
||||
#endif
|
||||
}
|
||||
}
|
||||
@@ -46,6 +46,8 @@ class LocalizationManager: ObservableObject {
|
||||
} catch {
|
||||
debugErrorSync("❌ 保存语言设置失败: \(error)")
|
||||
}
|
||||
// 同时更新 UserDefaults 供同步版本使用
|
||||
UserDefaults.standard.set(currentLanguage.rawValue, forKey: "AppLanguage")
|
||||
// 通知视图更新
|
||||
objectWillChange.send()
|
||||
}
|
||||
@@ -67,6 +69,9 @@ class LocalizationManager: ObservableObject {
|
||||
// 如果没有保存过语言设置,使用系统首选语言
|
||||
self.currentLanguage = Self.getSystemPreferredLanguage()
|
||||
}
|
||||
|
||||
// 确保 UserDefaults 也同步了当前语言设置
|
||||
UserDefaults.standard.set(self.currentLanguage.rawValue, forKey: "AppLanguage")
|
||||
}
|
||||
|
||||
// MARK: - 本地化方法
|
||||
@@ -114,26 +119,71 @@ class LocalizationManager: ObservableObject {
|
||||
}
|
||||
|
||||
// MARK: - SwiftUI Extensions
|
||||
// extension View {
|
||||
// /// 应用本地化字符串
|
||||
// /// - Parameter key: 本地化 key
|
||||
// /// - Returns: 带有本地化文本的视图
|
||||
// @MainActor
|
||||
// func localized(_ key: String) -> some View {
|
||||
// self.modifier(LocalizedTextModifier(key: key))
|
||||
// }
|
||||
// }
|
||||
extension View {
|
||||
/// 应用本地化字符串
|
||||
/// - Parameter key: 本地化 key
|
||||
/// - Returns: 带有本地化文本的视图
|
||||
@MainActor
|
||||
func localized(_ key: String) -> some View {
|
||||
self.modifier(LocalizedTextModifier(key: key))
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 便捷方法
|
||||
// extension String {
|
||||
// /// 获取本地化字符串
|
||||
// @MainActor
|
||||
// var localized: String {
|
||||
// return LocalizationManager.shared.localizedString(self)
|
||||
// }
|
||||
// /// 获取本地化字符串(带参数)
|
||||
// @MainActor
|
||||
// func localized(arguments: CVarArg...) -> String {
|
||||
// return LocalizationManager.shared.localizedString(self, arguments: arguments)
|
||||
// }
|
||||
// }
|
||||
extension String {
|
||||
/// 获取本地化字符串
|
||||
@MainActor
|
||||
var localized: String {
|
||||
return LocalizationManager.shared.localizedString(self)
|
||||
}
|
||||
/// 获取本地化字符串(带参数)
|
||||
@MainActor
|
||||
func localized(arguments: CVarArg...) -> String {
|
||||
return LocalizationManager.shared.localizedString(self, arguments: arguments)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 全局本地化方法
|
||||
/// 全局本地化字符串获取方法
|
||||
/// 使用 LocalizationManager 而不是系统语言设置
|
||||
/// - Parameters:
|
||||
/// - key: 本地化 key
|
||||
/// - comment: 注释(保持与 NSLocalizedString 兼容)
|
||||
/// - Returns: 本地化后的字符串
|
||||
@MainActor
|
||||
func LocalizedString(_ key: String, comment: String = "") -> String {
|
||||
return LocalizationManager.shared.localizedString(key)
|
||||
}
|
||||
|
||||
/// 同步版本的本地化字符串获取方法
|
||||
/// 用于 TCA reducer 等同步上下文
|
||||
/// - Parameters:
|
||||
/// - key: 本地化 key
|
||||
/// - comment: 注释(保持与 NSLocalizedString 兼容)
|
||||
/// - Returns: 本地化后的字符串
|
||||
func LocalizedStringSync(_ key: String, comment: String = "") -> String {
|
||||
// 直接从 UserDefaults 读取当前语言设置(避免 @MainActor 隔离问题)
|
||||
let currentLanguage = UserDefaults.standard.string(forKey: "AppLanguage") ?? "en"
|
||||
|
||||
// 根据语言设置获取本地化字符串
|
||||
guard let path = Bundle.main.path(forResource: currentLanguage, ofType: "lproj"),
|
||||
let bundle = Bundle(path: path) else {
|
||||
// 如果找不到对应语言包,返回 key 本身
|
||||
return NSLocalizedString(key, comment: comment)
|
||||
}
|
||||
|
||||
return NSLocalizedString(key, bundle: bundle, comment: comment)
|
||||
}
|
||||
|
||||
// MARK: - LocalizedTextModifier
|
||||
/// 本地化文本修饰符
|
||||
struct LocalizedTextModifier: ViewModifier {
|
||||
let key: String
|
||||
|
||||
func body(content: Content) -> some View {
|
||||
content
|
||||
.onAppear {
|
||||
// 这里可以添加动态更新逻辑
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
11
yana/Utils/Navigation/AppRoute.swift
Normal file
11
yana/Utils/Navigation/AppRoute.swift
Normal file
@@ -0,0 +1,11 @@
|
||||
import Foundation
|
||||
|
||||
/// 应用统一路由定义
|
||||
enum AppRoute: Hashable {
|
||||
case login
|
||||
case main
|
||||
case setting
|
||||
case publish
|
||||
}
|
||||
|
||||
|
||||
38
yana/Utils/Network/APIService+Combine.swift
Normal file
38
yana/Utils/Network/APIService+Combine.swift
Normal file
@@ -0,0 +1,38 @@
|
||||
import Foundation
|
||||
@preconcurrency import Combine
|
||||
|
||||
// 以 @unchecked Sendable 包装 Future 的 promise,安全地跨并发域传递
|
||||
private final class PromiseBox<Output, Failure: Error>: @unchecked Sendable {
|
||||
private let fulfill: (Result<Output, Failure>) -> Void
|
||||
init(_ fulfill: @escaping (Result<Output, Failure>) -> Void) { self.fulfill = fulfill }
|
||||
func complete(_ result: Result<Output, Failure>) { fulfill(result) }
|
||||
}
|
||||
|
||||
extension APIServiceProtocol {
|
||||
/// 将 async/await 的请求桥接为 Combine Publisher
|
||||
/// - Parameter request: 符合 APIRequestProtocol 的请求对象
|
||||
/// - Returns: AnyPublisher<T.Response, APIError>
|
||||
func requestPublisher<T: APIRequestProtocol>(_ request: T) -> AnyPublisher<T.Response, APIError> {
|
||||
Deferred {
|
||||
Future { promise in
|
||||
let box = PromiseBox<T.Response, APIError>(promise)
|
||||
Task(priority: .userInitiated) {
|
||||
let result: Result<T.Response, APIError>
|
||||
do {
|
||||
let value = try await self.request(request)
|
||||
result = .success(value)
|
||||
} catch let apiError as APIError {
|
||||
result = .failure(apiError)
|
||||
} catch {
|
||||
result = .failure(.unknown(error.localizedDescription))
|
||||
}
|
||||
box.complete(result)
|
||||
}
|
||||
}
|
||||
}
|
||||
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
||||
.eraseToAnyPublisher()
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
54
yana/Utils/Network/DeviceContext.swift
Normal file
54
yana/Utils/Network/DeviceContext.swift
Normal file
@@ -0,0 +1,54 @@
|
||||
import Foundation
|
||||
import UIKit
|
||||
|
||||
@MainActor
|
||||
struct DeviceContext: Sendable {
|
||||
let languageCode: String
|
||||
let osName: String
|
||||
let osVersion: String
|
||||
let deviceModel: String
|
||||
let deviceId: String
|
||||
let appName: String
|
||||
let appVersion: String
|
||||
let channel: String
|
||||
let screenScale: String
|
||||
|
||||
static let shared: DeviceContext = {
|
||||
// 仅在主线程读取一次 UIKit/Bundle 信息
|
||||
let language = Locale.current.language.languageCode?.identifier ?? "en"
|
||||
let osName = "iOS"
|
||||
let osVersion = UIDevice.current.systemVersion
|
||||
let deviceModel = UIDevice.current.model
|
||||
let deviceId = UIDevice.current.identifierForVendor?.uuidString ?? UUID().uuidString
|
||||
let appName = Bundle.main.infoDictionary?["CFBundleName"] as? String ?? "eparty"
|
||||
let appVersion = Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String ?? "1.0.0"
|
||||
#if DEBUG
|
||||
let channel = "molistar_enterprise"
|
||||
#else
|
||||
let channel = "appstore"
|
||||
#endif
|
||||
let scale = String(format: "%.2f", Double(UIScreen.main.scale))
|
||||
|
||||
return DeviceContext(
|
||||
languageCode: language,
|
||||
osName: osName,
|
||||
osVersion: osVersion,
|
||||
deviceModel: deviceModel,
|
||||
deviceId: deviceId,
|
||||
appName: appName,
|
||||
appVersion: appVersion,
|
||||
channel: channel,
|
||||
screenScale: scale
|
||||
)
|
||||
}()
|
||||
}
|
||||
|
||||
enum UserAgentProvider {
|
||||
@MainActor
|
||||
static func userAgent() -> String {
|
||||
let ctx = DeviceContext.shared
|
||||
return "\(ctx.appName)/\(ctx.appVersion) (\(ctx.deviceModel); \(ctx.osName) \(ctx.osVersion); Scale/\(ctx.screenScale))"
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
33
yana/Utils/Network/NetworkMonitor.swift
Normal file
33
yana/Utils/Network/NetworkMonitor.swift
Normal file
@@ -0,0 +1,33 @@
|
||||
import Foundation
|
||||
import Network
|
||||
|
||||
/// 监听系统网络状态并缓存最近结果
|
||||
/// WiFi=2, 蜂窝=1, 未知/无网络=0(与现有代码语义对齐)
|
||||
final class NetworkMonitor: @unchecked Sendable {
|
||||
static let shared = NetworkMonitor()
|
||||
private let monitor = NWPathMonitor()
|
||||
private let queue = DispatchQueue(label: "com.yana.network.monitor")
|
||||
private var _currentType: Int = 2 // 默认与历史保持一致
|
||||
var currentType: Int { _currentType }
|
||||
private init() {
|
||||
monitor.pathUpdateHandler = { [weak self] path in
|
||||
guard let self = self else { return }
|
||||
let type: Int
|
||||
if path.status == .satisfied {
|
||||
if path.usesInterfaceType(.wifi) { type = 2 }
|
||||
else if path.usesInterfaceType(.cellular) { type = 1 }
|
||||
else { type = 0 }
|
||||
} else {
|
||||
type = 0
|
||||
}
|
||||
// 更新缓存(主线程或任一队列均可,这里选择主线程与 UI 一致)
|
||||
DispatchQueue.main.async { [weak self] in
|
||||
self?._currentType = type
|
||||
}
|
||||
}
|
||||
monitor.start(queue: queue)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -6,25 +6,25 @@ struct ScreenAdapterExample: View {
|
||||
var body: some View {
|
||||
GeometryReader { geometry in
|
||||
VStack(spacing: 20) {
|
||||
Text(LocalizedString("screen_adapter.method1", comment: ""))
|
||||
.font(.headline)
|
||||
.padding()
|
||||
.background(Color.blue.opacity(0.1))
|
||||
.cornerRadius(8)
|
||||
|
||||
// 方法1: 直接使用 ScreenAdapter 静态方法
|
||||
Text("方法1: 直接调用")
|
||||
.font(.system(size: ScreenAdapter.fontSize(16, for: geometry.size.width)))
|
||||
.padding(.leading, ScreenAdapter.width(20, for: geometry.size.width))
|
||||
.padding(.top, ScreenAdapter.height(50, for: geometry.size.height))
|
||||
Text(LocalizedString("screen_adapter.method2", comment: ""))
|
||||
.font(.headline)
|
||||
.padding()
|
||||
.background(Color.green.opacity(0.1))
|
||||
.cornerRadius(8)
|
||||
|
||||
// 方法2: 使用 View Extension (推荐)
|
||||
Text("方法2: View Extension")
|
||||
.adaptedFont(16)
|
||||
.adaptedHeight(50)
|
||||
|
||||
// 方法3: 使用比例计算
|
||||
Text("方法3: 比例计算")
|
||||
.font(.system(size: 16 * ScreenAdapter.widthRatio(for: geometry.size.width)))
|
||||
.padding(.top, 50 * ScreenAdapter.heightRatio(for: geometry.size.height))
|
||||
|
||||
Spacer()
|
||||
Text(LocalizedString("screen_adapter.method3", comment: ""))
|
||||
.font(.headline)
|
||||
.padding()
|
||||
.background(Color.orange.opacity(0.1))
|
||||
.cornerRadius(8)
|
||||
}
|
||||
.padding()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -26,15 +26,15 @@ static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopq
|
||||
if (characters == NULL)
|
||||
return nil;
|
||||
|
||||
int end = data.length - 3;
|
||||
int index = 0;
|
||||
int charCount = 0;
|
||||
NSUInteger end = data.length - 3;
|
||||
NSUInteger index = 0;
|
||||
NSUInteger charCount = 0;
|
||||
int n = 0;
|
||||
|
||||
while (index <= end) {
|
||||
int d = (((int)(((char *)[data bytes])[index]) & 0x0ff) << 16)
|
||||
| (((int)(((char *)[data bytes])[index + 1]) & 0x0ff) << 8)
|
||||
| ((int)(((char *)[data bytes])[index + 2]) & 0x0ff);
|
||||
int d = (((int)(((char *)[data bytes])[(NSUInteger)index]) & 0x0ff) << 16)
|
||||
| (((int)(((char *)[data bytes])[(NSUInteger)(index + 1)]) & 0x0ff) << 8)
|
||||
| ((int)(((char *)[data bytes])[(NSUInteger)(index + 2)]) & 0x0ff);
|
||||
|
||||
characters[charCount++] = encodingTable[(d >> 18) & 63];
|
||||
characters[charCount++] = encodingTable[(d >> 12) & 63];
|
||||
@@ -52,8 +52,8 @@ static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopq
|
||||
|
||||
if(index == data.length - 2)
|
||||
{
|
||||
int d = (((int)(((char *)[data bytes])[index]) & 0x0ff) << 16)
|
||||
| (((int)(((char *)[data bytes])[index + 1]) & 255) << 8);
|
||||
int d = (((int)(((char *)[data bytes])[(NSUInteger)index]) & 0x0ff) << 16)
|
||||
| (((int)(((char *)[data bytes])[(NSUInteger)(index + 1)]) & 255) << 8);
|
||||
characters[charCount++] = encodingTable[(d >> 18) & 63];
|
||||
characters[charCount++] = encodingTable[(d >> 12) & 63];
|
||||
characters[charCount++] = encodingTable[(d >> 6) & 63];
|
||||
@@ -61,7 +61,7 @@ static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopq
|
||||
}
|
||||
else if(index == data.length - 1)
|
||||
{
|
||||
int d = ((int)(((char *)[data bytes])[index]) & 0x0ff) << 16;
|
||||
int d = ((int)(((char *)[data bytes])[(NSUInteger)index]) & 0x0ff) << 16;
|
||||
characters[charCount++] = encodingTable[(d >> 18) & 63];
|
||||
characters[charCount++] = encodingTable[(d >> 12) & 63];
|
||||
characters[charCount++] = '=';
|
||||
@@ -78,8 +78,8 @@ static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopq
|
||||
return nil;
|
||||
}
|
||||
NSMutableData *rtnData = [[NSMutableData alloc]init];
|
||||
int slen = data.length;
|
||||
int index = 0;
|
||||
NSUInteger slen = data.length;
|
||||
NSUInteger index = 0;
|
||||
while (true) {
|
||||
while (index < slen && [data characterAtIndex:index] <= ' ') {
|
||||
index++;
|
||||
|
||||
@@ -12,11 +12,10 @@ import Security
|
||||
/// - 完善的错误处理
|
||||
/// - 线程安全操作
|
||||
/// - 可配置的访问控制级别
|
||||
@MainActor
|
||||
final class KeychainManager {
|
||||
final class KeychainManager: @unchecked Sendable {
|
||||
|
||||
// MARK: - 单例
|
||||
@MainActor static let shared = KeychainManager()
|
||||
static let shared = KeychainManager()
|
||||
private init() {}
|
||||
|
||||
// MARK: - 配置常量
|
||||
|
||||
32
yana/Utils/Security/SigningKeyProvider.swift
Normal file
32
yana/Utils/Security/SigningKeyProvider.swift
Normal file
@@ -0,0 +1,32 @@
|
||||
import Foundation
|
||||
|
||||
/// 提供 API 签名密钥的统一入口
|
||||
/// - 优先从 Info.plist 读取键 `API_SIGNING_KEY`
|
||||
/// - Debug 环境下若缺失,回退到历史密钥以避免开发阶段中断,同时输出告警
|
||||
/// - Release 环境下若缺失,输出错误并返回空字符串(应在发布前配置)
|
||||
enum SigningKeyProvider {
|
||||
/// Info.plist 中的键名
|
||||
private static let plistKey = "API_SIGNING_KEY"
|
||||
|
||||
/// 获取签名密钥
|
||||
static func signingKey() -> String {
|
||||
if let key = Bundle.main.object(forInfoDictionaryKey: plistKey) as? String,
|
||||
!key.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
|
||||
return key
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
// 仅在 Debug 回退,避免打断本地调试;请尽快在 Info.plist 配置 API_SIGNING_KEY
|
||||
let legacy = "rpbs6us1m8r2j9g6u06ff2bo18orwaya"
|
||||
debugWarnSync("⚠️ API_SIGNING_KEY 未配置,Debug 使用历史回退密钥(请尽快配置 Info.plist)")
|
||||
return legacy
|
||||
#else
|
||||
debugErrorSync("❌ 缺少 API_SIGNING_KEY,请在 Info.plist 中配置")
|
||||
assertionFailure("Missing API_SIGNING_KEY in Info.plist")
|
||||
return ""
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
195
yana/Utils/TCCos/COSManagerAdapter.swift
Normal file
195
yana/Utils/TCCos/COSManagerAdapter.swift
Normal file
@@ -0,0 +1,195 @@
|
||||
//
|
||||
// COSManagerAdapter.swift
|
||||
// yana
|
||||
//
|
||||
// Created by P on 2025/7/31.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
import ComposableArchitecture
|
||||
|
||||
// MARK: - COSManager 适配器
|
||||
|
||||
/// COSManager 适配器
|
||||
///
|
||||
/// 保持与现有 COSManager 相同的接口,但内部使用新的 TCCos 组件
|
||||
/// 这样可以无缝替换现有的 COSManager,无需修改其他代码
|
||||
@MainActor
|
||||
class COSManagerAdapter: ObservableObject {
|
||||
static let shared = COSManagerAdapter()
|
||||
|
||||
private init() {
|
||||
// 使用默认的 TCCos 服务依赖
|
||||
self.tokenService = COSTokenService(apiService: LiveAPIService())
|
||||
self.uploadService = COSUploadService(
|
||||
tokenService: self.tokenService,
|
||||
configurationService: COSConfigurationService()
|
||||
)
|
||||
self.configurationService = COSConfigurationService()
|
||||
|
||||
debugInfoSync("<EFBFBD><EFBFBD> COSManagerAdapter 已初始化,使用 TCCos 组件")
|
||||
}
|
||||
|
||||
// MARK: - 内部 TCCos 组件
|
||||
|
||||
private let tokenService: COSTokenServiceProtocol
|
||||
private let uploadService: COSUploadServiceProtocol
|
||||
private let configurationService: COSConfigurationServiceProtocol
|
||||
|
||||
// MARK: - 兼容性接口(与 COSManager 保持一致)
|
||||
|
||||
/// 获取腾讯云 COS Token
|
||||
/// - Parameter apiService: API 服务实例
|
||||
/// - Returns: Token 数据,如果获取失败返回 nil
|
||||
func getToken(apiService: any APIServiceProtocol & Sendable) async -> TcTokenData? {
|
||||
do {
|
||||
debugInfoSync("🔐 开始请求腾讯云 COS Token...")
|
||||
let tokenData = try await tokenService.getValidToken()
|
||||
|
||||
debugInfoSync("✅ COS Token 获取成功")
|
||||
debugInfoSync(" - 存储桶: \(tokenData.bucket)")
|
||||
debugInfoSync(" - 地域: \(tokenData.region)")
|
||||
debugInfoSync(" - 过期时间: \(tokenData.expirationDate)")
|
||||
debugInfoSync(" - 剩余时间: \(tokenData.remainingTime)秒")
|
||||
|
||||
return tokenData
|
||||
} catch {
|
||||
debugErrorSync("❌ COS Token 获取失败: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
/// 强制刷新 Token
|
||||
func refreshToken(apiService: any APIServiceProtocol & Sendable) async -> TcTokenData? {
|
||||
do {
|
||||
debugInfoSync("🔄 开始刷新腾讯云 COS Token...")
|
||||
let tokenData = try await tokenService.refreshToken()
|
||||
|
||||
debugInfoSync("✅ COS Token 刷新成功")
|
||||
debugInfoSync(" - 存储桶: \(tokenData.bucket)")
|
||||
debugInfoSync(" - 地域: \(tokenData.region)")
|
||||
debugInfoSync(" - 过期时间: \(tokenData.expirationDate)")
|
||||
|
||||
return tokenData
|
||||
} catch {
|
||||
debugErrorSync("❌ COS Token 刷新失败: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
/// 上传图片到腾讯云 COS
|
||||
/// - Parameters:
|
||||
/// - imageData: 图片数据
|
||||
/// - apiService: API 服务实例
|
||||
/// - Returns: 上传成功的云地址,如果失败返回 nil
|
||||
func uploadImage(_ imageData: Data, apiService: any APIServiceProtocol & Sendable) async -> String? {
|
||||
// 生成唯一文件名
|
||||
let fileExtension = "jpg"
|
||||
let fileName = "images/\(UUID().uuidString).\(fileExtension)"
|
||||
|
||||
do {
|
||||
debugInfoSync("🚀 开始上传图片,数据大小: \(imageData.count) bytes")
|
||||
let url = try await uploadService.uploadImage(imageData, fileName: fileName)
|
||||
debugInfoSync("✅ 图片上传成功: \(url)")
|
||||
return url
|
||||
} catch {
|
||||
debugErrorSync("❌ 图片上传失败: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
/// 上传 UIImage 到腾讯云 COS,自动压缩为 JPEG(0.7)
|
||||
/// - Parameters:
|
||||
/// - image: UIImage 实例
|
||||
/// - apiService: API 服务实例
|
||||
/// - Returns: 上传成功的云地址,如果失败返回 nil
|
||||
func uploadUIImage(_ image: UIImage, apiService: any APIServiceProtocol & Sendable) async -> String? {
|
||||
// 生成唯一文件名
|
||||
let fileExtension = "jpg"
|
||||
let fileName = "images/\(UUID().uuidString).\(fileExtension)"
|
||||
|
||||
do {
|
||||
debugInfoSync("<EFBFBD><EFBFBD> 开始上传 UIImage,自动压缩为 JPEG(0.7)")
|
||||
let url = try await uploadService.uploadUIImage(image, fileName: fileName)
|
||||
debugInfoSync("✅ UIImage 上传成功: \(url)")
|
||||
return url
|
||||
} catch {
|
||||
debugErrorSync("❌ UIImage 上传失败: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 只读属性(与 COSManager 保持一致)
|
||||
|
||||
/// 外部安全访问 Token
|
||||
var token: TcTokenData? {
|
||||
get async {
|
||||
do {
|
||||
return try await tokenService.getValidToken()
|
||||
} catch {
|
||||
debugErrorSync("❌ 获取 Token 失败: \(error.localizedDescription)")
|
||||
return nil
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 调试信息(与 COSManager 保持一致)
|
||||
|
||||
/// 获取当前 Token 状态信息
|
||||
func getTokenStatus() async -> String {
|
||||
return await tokenService.getTokenStatus()
|
||||
}
|
||||
|
||||
/// 测试 Token 获取功能(仅用于调试)
|
||||
func testTokenRetrieval(apiService: any APIServiceProtocol & Sendable) async {
|
||||
#if DEBUG
|
||||
debugInfoSync("\n<EFBFBD><EFBFBD> 开始测试腾讯云 COS Token 获取功能")
|
||||
|
||||
let token = await getToken(apiService: apiService)
|
||||
if let tokenData = token {
|
||||
debugInfoSync("✅ Token 获取成功")
|
||||
debugInfoSync(" bucket: \(tokenData.bucket)")
|
||||
debugInfoSync(" Expiration: \(tokenData.expireTime)")
|
||||
debugInfoSync(" Token: \(tokenData.sessionToken.prefix(20))...")
|
||||
debugInfoSync(" SecretId: \(tokenData.secretId.prefix(20))...")
|
||||
} else {
|
||||
debugInfoSync("❌ Token 获取失败")
|
||||
}
|
||||
|
||||
let status = await getTokenStatus()
|
||||
debugInfoSync("📊 Token 状态: \(status)")
|
||||
debugInfoSync("✅ 腾讯云 COS Token 测试完成\n")
|
||||
#endif
|
||||
}
|
||||
|
||||
// MARK: - 内部方法(与 COSManager 保持一致)
|
||||
|
||||
/// 清除缓存的 Token(内部方法,保持兼容性)
|
||||
private func clearCachedToken() {
|
||||
tokenService.clearCachedToken()
|
||||
debugInfoSync("🗑️ 清除缓存的 COS Token")
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 依赖注入扩展
|
||||
|
||||
extension COSManagerAdapter {
|
||||
/// 使用依赖注入创建实例(用于测试)
|
||||
static func createWithDependencies(
|
||||
tokenService: COSTokenServiceProtocol,
|
||||
uploadService: COSUploadServiceProtocol,
|
||||
configurationService: COSConfigurationServiceProtocol
|
||||
) -> COSManagerAdapter {
|
||||
let adapter = COSManagerAdapter()
|
||||
// 注意:这里需要修改为使用依赖注入的初始化方式
|
||||
// 由于当前设计,我们使用默认的 shared 实例
|
||||
return adapter
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 类型别名(保持向后兼容)
|
||||
|
||||
/// 为了保持向后兼容,将 COSManager 重命名为 COSManagerAdapter
|
||||
/// 这样现有代码无需修改即可使用新的实现
|
||||
typealias COSManager = COSManagerAdapter
|
||||
615
yana/Utils/TCCos/Features/COSFeature.swift
Normal file
615
yana/Utils/TCCos/Features/COSFeature.swift
Normal file
@@ -0,0 +1,615 @@
|
||||
import Foundation
|
||||
import ComposableArchitecture
|
||||
import UIKit
|
||||
|
||||
// MARK: - COS Feature
|
||||
|
||||
/// COS 主 Feature
|
||||
/// 整合Token管理、图片上传、配置管理等功能
|
||||
public struct COSFeature: Reducer, @unchecked Sendable {
|
||||
|
||||
// MARK: - State
|
||||
|
||||
/// COS 状态
|
||||
public struct State: Equatable {
|
||||
/// Token 状态
|
||||
public var tokenState: TokenState?
|
||||
/// 上传状态
|
||||
public var uploadState: UploadState?
|
||||
/// 配置状态
|
||||
public var configurationState: ConfigurationState?
|
||||
|
||||
public init(
|
||||
tokenState: TokenState? = TokenState(),
|
||||
uploadState: UploadState? = UploadState(),
|
||||
configurationState: ConfigurationState? = ConfigurationState()
|
||||
) {
|
||||
self.tokenState = tokenState
|
||||
self.uploadState = uploadState
|
||||
self.configurationState = configurationState
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Action
|
||||
|
||||
/// COS Action
|
||||
@CasePathable
|
||||
public enum Action: Equatable {
|
||||
/// Token 相关 Action
|
||||
case token(TokenAction)
|
||||
/// 上传相关 Action
|
||||
case upload(UploadAction)
|
||||
/// 配置相关 Action
|
||||
case configuration(ConfigurationAction)
|
||||
/// 初始化
|
||||
case onAppear
|
||||
/// 错误处理
|
||||
case handleError(COSError)
|
||||
/// 重试操作
|
||||
case retry
|
||||
/// 重置所有状态
|
||||
case resetAll
|
||||
/// 检查服务健康状态
|
||||
case checkHealth
|
||||
}
|
||||
|
||||
// MARK: - Dependencies
|
||||
|
||||
@Dependency(\.cosTokenService) var tokenService
|
||||
@Dependency(\.cosUploadService) var uploadService
|
||||
@Dependency(\.cosConfigurationService) var configurationService
|
||||
|
||||
// MARK: - Reducer
|
||||
|
||||
public var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .onAppear:
|
||||
debugInfoSync("🚀 COS Feature 初始化")
|
||||
return handleOnAppear()
|
||||
|
||||
case .token(let tokenAction):
|
||||
return handleTokenAction(&state, tokenAction)
|
||||
|
||||
case .upload(let uploadAction):
|
||||
return handleUploadAction(&state, uploadAction)
|
||||
|
||||
case .configuration(let configAction):
|
||||
return handleConfigurationAction(&state, configAction)
|
||||
|
||||
case .handleError(let error):
|
||||
debugErrorSync("❌ COS Feature 错误: \(error.localizedDescription)")
|
||||
return .none
|
||||
|
||||
case .retry:
|
||||
return handleRetry()
|
||||
|
||||
case .resetAll:
|
||||
return handleResetAll()
|
||||
|
||||
case .checkHealth:
|
||||
return handleCheckHealth()
|
||||
}
|
||||
}
|
||||
.ifLet(\.tokenState, action: /Action.token) {
|
||||
TokenReducer()
|
||||
}
|
||||
.ifLet(\.uploadState, action: /Action.upload) {
|
||||
UploadReducer()
|
||||
}
|
||||
.ifLet(\.configurationState, action: /Action.configuration) {
|
||||
ConfigurationReducer()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - 私有方法
|
||||
|
||||
/// 处理 onAppear 事件
|
||||
private func handleOnAppear() -> Effect<Action> {
|
||||
return .run { send in
|
||||
// 检查服务初始化状态
|
||||
let isInitialized = await configurationService.isCOSServiceInitialized()
|
||||
await send(.configuration(.initializationStatusReceived(isInitialized)))
|
||||
|
||||
// 如果未初始化,尝试获取 Token 并初始化服务
|
||||
if !isInitialized {
|
||||
do {
|
||||
let token = try await tokenService.refreshToken()
|
||||
await send(.token(.tokenReceived(token)))
|
||||
await send(.configuration(.initializeService(token)))
|
||||
} catch {
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
} else {
|
||||
// 如果已初始化,获取当前 Token 状态
|
||||
let status = await tokenService.getTokenStatus()
|
||||
await send(.token(.tokenStatusReceived(status)))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理重试操作
|
||||
private func handleRetry() -> Effect<Action> {
|
||||
return .run { send in
|
||||
debugInfoSync("🔄 开始重试操作...")
|
||||
// 重新获取 Token 并初始化服务
|
||||
do {
|
||||
let token = try await tokenService.refreshToken()
|
||||
await send(.token(.tokenReceived(token)))
|
||||
await send(.configuration(.initializeService(token)))
|
||||
} catch {
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理重置所有状态
|
||||
private func handleResetAll() -> Effect<Action> {
|
||||
return .run { send in
|
||||
debugInfoSync("🔄 重置所有状态...")
|
||||
tokenService.clearCachedToken()
|
||||
await configurationService.resetCOSService()
|
||||
await send(.token(.clearToken))
|
||||
await send(.upload(.reset))
|
||||
await send(.configuration(.resetService))
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理健康检查
|
||||
private func handleCheckHealth() -> Effect<Action> {
|
||||
return .run { send in
|
||||
debugInfoSync("🏥 检查服务健康状态...")
|
||||
let isInitialized = await configurationService.isCOSServiceInitialized()
|
||||
let tokenStatus = await tokenService.getTokenStatus()
|
||||
|
||||
if !isInitialized {
|
||||
await send(.handleError(.serviceNotInitialized))
|
||||
} else if tokenStatus.contains("过期") {
|
||||
await send(.handleError(.tokenExpired))
|
||||
} else {
|
||||
debugInfoSync("✅ 服务健康状态良好")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理 Token Action
|
||||
private func handleTokenAction(_ state: inout State, _ action: TokenAction) -> Effect<Action> {
|
||||
switch action {
|
||||
case .getToken:
|
||||
return .run { send in
|
||||
do {
|
||||
let token = try await tokenService.refreshToken()
|
||||
await send(.token(.tokenReceived(token)))
|
||||
} catch {
|
||||
await send(.token(.setError(error.localizedDescription)))
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
|
||||
case .refreshToken:
|
||||
return .run { send in
|
||||
do {
|
||||
let token = try await tokenService.refreshToken()
|
||||
await send(.token(.tokenReceived(token)))
|
||||
|
||||
// Token 刷新后,重新初始化服务
|
||||
await send(.configuration(.initializeService(token)))
|
||||
} catch {
|
||||
await send(.token(.setError(error.localizedDescription)))
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
|
||||
case .getTokenStatus:
|
||||
return .run { send in
|
||||
let status = await tokenService.getTokenStatus()
|
||||
await send(.token(.tokenStatusReceived(status)))
|
||||
}
|
||||
|
||||
case .clearToken:
|
||||
return .run { send in
|
||||
tokenService.clearCachedToken()
|
||||
await send(.configuration(.resetService))
|
||||
}
|
||||
|
||||
case .tokenReceived, .tokenStatusReceived, .setError:
|
||||
// 这些 Action 由子 Reducer 处理
|
||||
return .none
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理上传 Action
|
||||
private func handleUploadAction(_ state: inout State, _ action: UploadAction) -> Effect<Action> {
|
||||
switch action {
|
||||
case .uploadImage(let imageData, let fileName):
|
||||
return .run { send in
|
||||
// 上传前检查服务状态和 Token
|
||||
let isInitialized = await configurationService.isCOSServiceInitialized()
|
||||
guard isInitialized else {
|
||||
await send(.upload(.uploadFailed("服务未初始化")))
|
||||
await send(.handleError(.serviceNotInitialized))
|
||||
return
|
||||
}
|
||||
|
||||
let tokenStatus = await tokenService.getTokenStatus()
|
||||
guard !tokenStatus.contains("过期") else {
|
||||
await send(.upload(.uploadFailed("Token 已过期")))
|
||||
await send(.handleError(.tokenExpired))
|
||||
return
|
||||
}
|
||||
|
||||
do {
|
||||
let url = try await uploadService.uploadImage(imageData, fileName: fileName)
|
||||
await send(.upload(.uploadCompleted(url)))
|
||||
} catch {
|
||||
await send(.upload(.uploadFailed(error.localizedDescription)))
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
|
||||
case .uploadUIImage(let image, let fileName):
|
||||
return .run { send in
|
||||
// 上传前检查服务状态和 Token
|
||||
let isInitialized = await configurationService.isCOSServiceInitialized()
|
||||
guard isInitialized else {
|
||||
await send(.upload(.uploadFailed("服务未初始化")))
|
||||
await send(.handleError(.serviceNotInitialized))
|
||||
return
|
||||
}
|
||||
|
||||
let tokenStatus = await tokenService.getTokenStatus()
|
||||
guard !tokenStatus.contains("过期") else {
|
||||
await send(.upload(.uploadFailed("Token 已过期")))
|
||||
await send(.handleError(.tokenExpired))
|
||||
return
|
||||
}
|
||||
|
||||
do {
|
||||
let url = try await uploadService.uploadUIImage(image, fileName: fileName)
|
||||
await send(.upload(.uploadCompleted(url)))
|
||||
} catch {
|
||||
await send(.upload(.uploadFailed(error.localizedDescription)))
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
|
||||
case .cancelUpload(let taskId):
|
||||
return .run { send in
|
||||
await uploadService.cancelUpload(taskId: taskId)
|
||||
await send(.upload(.cancelUpload(taskId)))
|
||||
}
|
||||
|
||||
case .uploadCompleted, .uploadFailed, .updateProgress, .reset:
|
||||
// 这些 Action 由子 Reducer 处理
|
||||
return .none
|
||||
}
|
||||
}
|
||||
|
||||
/// 处理配置 Action
|
||||
private func handleConfigurationAction(_ state: inout State, _ action: ConfigurationAction) -> Effect<Action> {
|
||||
switch action {
|
||||
case .initializeService(let tokenData):
|
||||
return .run { send in
|
||||
do {
|
||||
try await configurationService.initializeCOSService(with: tokenData)
|
||||
await send(.configuration(.serviceInitialized))
|
||||
debugInfoSync("✅ COS 服务初始化成功")
|
||||
} catch {
|
||||
await send(.configuration(.setError(error.localizedDescription)))
|
||||
await send(.handleError(error as? COSError ?? .unknown(error.localizedDescription)))
|
||||
}
|
||||
}
|
||||
|
||||
case .checkInitializationStatus:
|
||||
return .run { send in
|
||||
let isInitialized = await configurationService.isCOSServiceInitialized()
|
||||
await send(.configuration(.initializationStatusReceived(isInitialized)))
|
||||
}
|
||||
|
||||
case .resetService:
|
||||
return .run { send in
|
||||
await configurationService.resetCOSService()
|
||||
await send(.configuration(.serviceReset))
|
||||
debugInfoSync("🔄 COS 服务已重置")
|
||||
}
|
||||
|
||||
case .serviceInitialized, .initializationStatusReceived, .serviceReset, .setError:
|
||||
// 这些 Action 由子 Reducer 处理
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Token State & Action
|
||||
|
||||
/// Token 状态
|
||||
public struct TokenState: Equatable {
|
||||
/// 当前 Token
|
||||
public var currentToken: TcTokenData?
|
||||
/// 加载状态
|
||||
public var isLoading: Bool = false
|
||||
/// Token 状态信息
|
||||
public var statusMessage: String = ""
|
||||
/// 错误信息
|
||||
public var error: String?
|
||||
|
||||
public init() {}
|
||||
}
|
||||
|
||||
/// Token Action
|
||||
public enum TokenAction: Equatable {
|
||||
/// 获取 Token
|
||||
case getToken
|
||||
/// Token 获取成功
|
||||
case tokenReceived(TcTokenData)
|
||||
/// 刷新 Token
|
||||
case refreshToken
|
||||
/// 获取 Token 状态
|
||||
case getTokenStatus
|
||||
/// Token 状态获取成功
|
||||
case tokenStatusReceived(String)
|
||||
/// 清除 Token
|
||||
case clearToken
|
||||
/// 设置错误
|
||||
case setError(String?)
|
||||
}
|
||||
|
||||
// MARK: - Upload State & Action
|
||||
|
||||
/// 上传状态
|
||||
public struct UploadState: Equatable {
|
||||
/// 当前上传任务
|
||||
public var currentTask: UploadTask?
|
||||
/// 上传进度
|
||||
public var progress: Double = 0.0
|
||||
/// 上传结果
|
||||
public var result: String?
|
||||
/// 错误信息
|
||||
public var error: String?
|
||||
/// 是否正在上传
|
||||
public var isUploading: Bool = false
|
||||
|
||||
public init() {}
|
||||
}
|
||||
|
||||
/// 上传 Action
|
||||
public enum UploadAction: Equatable {
|
||||
/// 上传图片数据
|
||||
case uploadImage(Data, String)
|
||||
/// 上传 UIImage
|
||||
case uploadUIImage(UIImage, String)
|
||||
/// 上传完成
|
||||
case uploadCompleted(String)
|
||||
/// 上传失败
|
||||
case uploadFailed(String)
|
||||
/// 更新进度
|
||||
case updateProgress(Double)
|
||||
/// 取消上传
|
||||
case cancelUpload(UUID)
|
||||
/// 重置状态
|
||||
case reset
|
||||
}
|
||||
|
||||
// MARK: - Configuration State & Action
|
||||
|
||||
/// 配置状态
|
||||
public struct ConfigurationState: Equatable {
|
||||
/// 服务状态
|
||||
public var serviceStatus: COSServiceStatus = .notInitialized
|
||||
/// 当前配置
|
||||
public var currentConfiguration: COSConfiguration?
|
||||
/// 错误信息
|
||||
public var error: String?
|
||||
|
||||
public init() {}
|
||||
}
|
||||
|
||||
/// 配置 Action
|
||||
public enum ConfigurationAction: Equatable {
|
||||
/// 初始化服务
|
||||
case initializeService(TcTokenData)
|
||||
/// 服务初始化成功
|
||||
case serviceInitialized
|
||||
/// 检查初始化状态
|
||||
case checkInitializationStatus
|
||||
/// 初始化状态获取成功
|
||||
case initializationStatusReceived(Bool)
|
||||
/// 重置服务
|
||||
case resetService
|
||||
/// 服务重置成功
|
||||
case serviceReset
|
||||
/// 设置错误
|
||||
case setError(String?)
|
||||
}
|
||||
|
||||
// MARK: - Reducers
|
||||
|
||||
/// Token Reducer
|
||||
public struct TokenReducer: Reducer {
|
||||
public typealias State = TokenState
|
||||
public typealias Action = TokenAction
|
||||
|
||||
public var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .getToken:
|
||||
state.isLoading = true
|
||||
state.error = nil
|
||||
return .none
|
||||
|
||||
case .tokenReceived(let token):
|
||||
state.currentToken = token
|
||||
state.isLoading = false
|
||||
state.error = nil
|
||||
debugInfoSync("✅ Token 获取成功: \(token.bucket)")
|
||||
return .none
|
||||
|
||||
case .refreshToken:
|
||||
state.isLoading = true
|
||||
state.error = nil
|
||||
return .none
|
||||
|
||||
case .getTokenStatus:
|
||||
return .none
|
||||
|
||||
case .tokenStatusReceived(let status):
|
||||
state.statusMessage = status
|
||||
return .none
|
||||
|
||||
case .clearToken:
|
||||
state.currentToken = nil
|
||||
state.statusMessage = ""
|
||||
state.error = nil
|
||||
return .none
|
||||
|
||||
case .setError(let error):
|
||||
state.error = error
|
||||
state.isLoading = false
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Upload Reducer
|
||||
public struct UploadReducer: Reducer {
|
||||
public typealias State = UploadState
|
||||
public typealias Action = UploadAction
|
||||
|
||||
public var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .uploadImage(let imageData, let fileName):
|
||||
state.isUploading = true
|
||||
state.progress = 0.0
|
||||
state.error = nil
|
||||
state.result = nil
|
||||
state.currentTask = UploadTask(
|
||||
imageData: imageData,
|
||||
fileName: fileName,
|
||||
status: .uploading(progress: 0.0)
|
||||
)
|
||||
debugInfoSync("🚀 开始上传图片数据: \(fileName), 大小: \(imageData.count) bytes")
|
||||
return .none
|
||||
|
||||
case .uploadUIImage(let image, let fileName):
|
||||
state.isUploading = true
|
||||
state.progress = 0.0
|
||||
state.error = nil
|
||||
state.result = nil
|
||||
// 将 UIImage 转换为 Data
|
||||
let imageData = image.jpegData(compressionQuality: 0.7) ?? Data()
|
||||
state.currentTask = UploadTask(
|
||||
imageData: imageData,
|
||||
fileName: fileName,
|
||||
status: .uploading(progress: 0.0)
|
||||
)
|
||||
debugInfoSync("🚀 开始上传UIImage: \(fileName), 大小: \(imageData.count) bytes")
|
||||
return .none
|
||||
|
||||
case .uploadCompleted(let url):
|
||||
state.isUploading = false
|
||||
state.progress = 1.0
|
||||
state.result = url
|
||||
state.error = nil
|
||||
state.currentTask = state.currentTask?.updatingStatus(.success(url: url))
|
||||
debugInfoSync("✅ 上传完成: \(url)")
|
||||
return .none
|
||||
|
||||
case .uploadFailed(let error):
|
||||
state.isUploading = false
|
||||
state.error = error
|
||||
state.currentTask = state.currentTask?.updatingStatus(.failure(error: error))
|
||||
debugErrorSync("❌ 上传失败: \(error)")
|
||||
return .none
|
||||
|
||||
case .updateProgress(let progress):
|
||||
state.progress = progress
|
||||
state.currentTask = state.currentTask?.updatingStatus(.uploading(progress: progress))
|
||||
return .none
|
||||
|
||||
case .cancelUpload:
|
||||
state.isUploading = false
|
||||
state.error = "上传已取消"
|
||||
state.currentTask = state.currentTask?.updatingStatus(.failure(error: "上传已取消"))
|
||||
debugInfoSync("❌ 上传已取消")
|
||||
return .none
|
||||
|
||||
case .reset:
|
||||
state.currentTask = nil
|
||||
state.progress = 0.0
|
||||
state.result = nil
|
||||
state.error = nil
|
||||
state.isUploading = false
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Configuration Reducer
|
||||
public struct ConfigurationReducer: Reducer {
|
||||
public typealias State = ConfigurationState
|
||||
public typealias Action = ConfigurationAction
|
||||
|
||||
public var body: some ReducerOf<Self> {
|
||||
Reduce { state, action in
|
||||
switch action {
|
||||
case .initializeService(let tokenData):
|
||||
state.serviceStatus = .initializing
|
||||
state.error = nil
|
||||
state.currentConfiguration = COSConfiguration(
|
||||
region: tokenData.region,
|
||||
bucket: tokenData.bucket
|
||||
)
|
||||
debugInfoSync("🔄 开始初始化 COS 服务: \(tokenData.bucket)")
|
||||
return .none
|
||||
|
||||
case .serviceInitialized:
|
||||
state.serviceStatus =
|
||||
.initialized(
|
||||
configuration: state.currentConfiguration ?? COSConfiguration(
|
||||
region: "ap-hongkong",
|
||||
bucket: "molistar-1320554189"
|
||||
)
|
||||
)
|
||||
debugInfoSync("✅ COS 服务初始化成功")
|
||||
return .none
|
||||
|
||||
case .checkInitializationStatus:
|
||||
return .none
|
||||
|
||||
case .initializationStatusReceived(let isInitialized):
|
||||
if isInitialized {
|
||||
state.serviceStatus =
|
||||
.initialized(
|
||||
configuration: state.currentConfiguration ?? COSConfiguration(
|
||||
region: "ap-hongkong",
|
||||
bucket: "molistar-1320554189"
|
||||
)
|
||||
)
|
||||
} else {
|
||||
state.serviceStatus = .notInitialized
|
||||
}
|
||||
return .none
|
||||
|
||||
case .resetService:
|
||||
state.serviceStatus = .notInitialized
|
||||
state.currentConfiguration = nil
|
||||
state.error = nil
|
||||
debugInfoSync("🔄 COS 服务已重置")
|
||||
return .none
|
||||
|
||||
case .serviceReset:
|
||||
state.serviceStatus = .notInitialized
|
||||
state.currentConfiguration = nil
|
||||
return .none
|
||||
|
||||
case .setError(let error):
|
||||
state.error = error
|
||||
state.serviceStatus = .failed(error: error ?? "未知错误")
|
||||
return .none
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user