feat: 国际化(i18n) en/ja/ko + App 内语言切换
主体:多语言支持(简体中文源 + 英/日/韩)
- 基础设施:Localizable.xcstrings(String Catalog,sourceLanguage=zh-Hans)
+ pbxproj developmentRegion/knownRegions 注册 en/ja/ko
- 全部硬编码 Locale("zh_CN") → Locale.current;中文 dateFormat → Date.FormatStyle(跟随系统)
- UI 中文字面量统一为 String(appLoc:)(显式绑定所选语言 bundle+locale,即时切换)
Text 字面量走环境 \.locale + Bundle 重定向
- 549 个 catalog key 全部 en/ja/ko 翻译完成(0 未翻译)
- App 内语言切换:我的 → 语言(LanguageManager + 即时生效,无需重启)
- 双用预设(症状/监测指标/慢病)本地化:static→computed 避免缓存
注:本提交为 WIP,一并打包了并行进行的功能模块
(HealthExport 健康导出、Security/Face ID 锁、DiaryAssist 日记 AI 辅助)
及 App 图标、CLAUDE.md、docs/scripts。
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
68
康康/Models/HealthExport.swift
Normal file
68
康康/Models/HealthExport.swift
Normal file
@@ -0,0 +1,68 @@
|
||||
import Foundation
|
||||
import SwiftData
|
||||
|
||||
/// 「导出身体档案」单条历史。一次成功生成 = 一条 HealthExport。
|
||||
///
|
||||
/// 与 Indicator/Report 等源记录之间用 `[String]` 弱关联(而不是 SwiftData
|
||||
/// 关系),这样源记录被永久删除时,历史导出仍保留为快照。
|
||||
///
|
||||
/// 属性写法与项目其他 @Model(Indicator/ChatTurn 等)对齐:
|
||||
/// 不在属性上写 default,所有默认值都在 `init` 里。
|
||||
@Model
|
||||
final class HealthExport {
|
||||
var prompt: String
|
||||
var content: String
|
||||
var createdAt: Date
|
||||
|
||||
// 引用回链(§3.3 RAG 引用,W3 再做点击跳转)
|
||||
var referencedIndicatorIDs: [String]
|
||||
var referencedReportIDs: [String]
|
||||
var referencedSymptomIDs: [String]
|
||||
var referencedDiaryIDs: [String]
|
||||
|
||||
// 意图抽取快照,供「重新生成」复用,不再二次抽意图
|
||||
var inferredTimeFromDate: Date?
|
||||
var inferredTimeToDate: Date?
|
||||
var inferredIntent: String?
|
||||
|
||||
// demo 卖点凭证
|
||||
/// 模型 tag,如 "Qwen3-1.7B-4bit"。截图能证明本地推理。
|
||||
var modelTag: String
|
||||
/// 末次 tok/s,对应 demo 卖点 #6 Live Activity 数据。
|
||||
var decodeRate: Double
|
||||
|
||||
init(prompt: String = "",
|
||||
content: String = "",
|
||||
createdAt: Date = .now,
|
||||
referencedIndicatorIDs: [String] = [],
|
||||
referencedReportIDs: [String] = [],
|
||||
referencedSymptomIDs: [String] = [],
|
||||
referencedDiaryIDs: [String] = [],
|
||||
inferredTimeFromDate: Date? = nil,
|
||||
inferredTimeToDate: Date? = nil,
|
||||
inferredIntent: String? = nil,
|
||||
modelTag: String = "Qwen3-1.7B-4bit",
|
||||
decodeRate: Double = 0) {
|
||||
self.prompt = prompt
|
||||
self.content = content
|
||||
self.createdAt = createdAt
|
||||
self.referencedIndicatorIDs = referencedIndicatorIDs
|
||||
self.referencedReportIDs = referencedReportIDs
|
||||
self.referencedSymptomIDs = referencedSymptomIDs
|
||||
self.referencedDiaryIDs = referencedDiaryIDs
|
||||
self.inferredTimeFromDate = inferredTimeFromDate
|
||||
self.inferredTimeToDate = inferredTimeToDate
|
||||
self.inferredIntent = inferredIntent
|
||||
self.modelTag = modelTag
|
||||
self.decodeRate = decodeRate
|
||||
}
|
||||
}
|
||||
|
||||
extension HealthExport {
|
||||
/// 列表 / strip 显示的 prompt 摘要(≤ 30 字 + ...)
|
||||
var promptPreview: String {
|
||||
let trimmed = prompt.trimmingCharacters(in: .whitespacesAndNewlines)
|
||||
if trimmed.count <= 30 { return trimmed }
|
||||
return trimmed.prefix(30) + "…"
|
||||
}
|
||||
}
|
||||
@@ -10,11 +10,11 @@ enum ReportType: String, Codable, CaseIterable {
|
||||
|
||||
var label: String {
|
||||
switch self {
|
||||
case .checkup: return "体检报告"
|
||||
case .lab: return "化验单"
|
||||
case .imaging: return "影像报告"
|
||||
case .prescription: return "处方"
|
||||
case .other: return "其他"
|
||||
case .checkup: return String(appLoc: "体检报告")
|
||||
case .lab: return String(appLoc: "化验单")
|
||||
case .imaging: return String(appLoc: "影像报告")
|
||||
case .prescription: return String(appLoc: "处方")
|
||||
case .other: return String(appLoc: "其他")
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -250,12 +250,12 @@ final class MetricReminder {
|
||||
var isEveryDay: Bool { Set(weekdays) == Set(1...7) }
|
||||
|
||||
var frequencyLabel: String {
|
||||
if !enabled { return "已关闭" }
|
||||
if isEveryDay { return "每天" }
|
||||
if weekdays.isEmpty { return "未选日" }
|
||||
let names = ["日", "一", "二", "三", "四", "五", "六"]
|
||||
if !enabled { return String(appLoc: "已关闭") }
|
||||
if isEveryDay { return String(appLoc: "每天") }
|
||||
if weekdays.isEmpty { return String(appLoc: "未选日") }
|
||||
let names = [String(appLoc: "日"), String(appLoc: "一"), String(appLoc: "二"), String(appLoc: "三"), String(appLoc: "四"), String(appLoc: "五"), String(appLoc: "六")]
|
||||
let sorted = weekdays.sorted()
|
||||
return "每周 " + sorted.map { names[$0 - 1] }.joined()
|
||||
return String(appLoc: "每周 ") + sorted.map { names[$0 - 1] }.joined()
|
||||
}
|
||||
|
||||
var timeLabel: String {
|
||||
|
||||
@@ -57,9 +57,9 @@ extension UserProfile {
|
||||
|
||||
var label: String {
|
||||
switch self {
|
||||
case .male: return "男"
|
||||
case .female: return "女"
|
||||
case .undisclosed: return "不愿透露"
|
||||
case .male: return String(appLoc: "男")
|
||||
case .female: return String(appLoc: "女")
|
||||
case .undisclosed: return String(appLoc: "不愿透露")
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -78,7 +78,7 @@ extension UserProfile {
|
||||
/// 给 ProfileCard 一行预览:"38岁 · 男 · 175cm · 68kg · A型"
|
||||
var summaryLine: String {
|
||||
var parts: [String] = []
|
||||
if let age { parts.append("\(age)岁") }
|
||||
if let age { parts.append(String(appLoc: "\(age)岁")) }
|
||||
if sex != .undisclosed { parts.append(sex.label) }
|
||||
if let h = heightCM { parts.append("\(h)cm") }
|
||||
if let w = weightKG {
|
||||
@@ -87,7 +87,7 @@ extension UserProfile {
|
||||
: String(format: "%.1fkg", w)
|
||||
parts.append(s)
|
||||
}
|
||||
if !bloodTypeRaw.isEmpty { parts.append("\(bloodTypeRaw)型") }
|
||||
if !bloodTypeRaw.isEmpty { parts.append(String(appLoc: "\(bloodTypeRaw)型")) }
|
||||
return parts.joined(separator: " · ")
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user