Files
kangkang/康康/Models/UserProfile.swift
link2026 39edc25dc1 refactor(profile,monitor): move height/weight from MonitorMetric to UserProfile
身高/体重对成人变化慢,作为 Profile 静态字段比每次录入 Indicator 更合适。

- MonitorMetric:6 case(从 8 减),删 .height / .weight
- UserProfile:加 weightKG: Double?(支持小数),加 bmi computed
- summaryLine 加体重段:'175cm · 68.5kg'(整数省小数)
- ProfileEditView basics 加 weight 行 + footer 显示 BMI + 分类(偏瘦/正常/超重/肥胖)
- IndicatorQuickSheet:删 .height 回写 Profile 的特殊逻辑
- UserProfileTests:+5 个(weight 字段、summaryLine 含 weight、BMI 计算)

兼容性:老 Indicator 里的 seriesKey 'weight' / 'height' 数据保留(SwiftData String?
不变),只是新录入路径走 Profile 不走 Indicator;Trends 仍能用 String seriesKey
查询历史(如果将来要展示老数据)。

测试:60 case pass / 0 fail / 0 warning。
2026-05-26 07:58:47 +08:00

118 lines
3.6 KiB
Swift

import Foundation
import SwiftData
@Model
final class UserProfile {
// 5
var birthYear: Int? // 1990,
var biologicalSexRaw: String // "" / "male" / "female"
var heightCM: Int?
var weightKG: Double? // (68.5)
var bloodTypeRaw: String // "" / "A" / "B" / "AB" / "O"
//
var allergies: [String]
var chronicConditions: [String]
var familyHistory: [String]
//
var currentMedications: [String]
var updatedAt: Date
init(birthYear: Int? = nil,
biologicalSexRaw: String = "",
heightCM: Int? = nil,
weightKG: Double? = nil,
bloodTypeRaw: String = "",
allergies: [String] = [],
chronicConditions: [String] = [],
familyHistory: [String] = [],
currentMedications: [String] = [],
updatedAt: Date = .now) {
self.birthYear = birthYear
self.biologicalSexRaw = biologicalSexRaw
self.heightCM = heightCM
self.weightKG = weightKG
self.bloodTypeRaw = bloodTypeRaw
self.allergies = allergies
self.chronicConditions = chronicConditions
self.familyHistory = familyHistory
self.currentMedications = currentMedications
self.updatedAt = updatedAt
}
}
extension UserProfile {
enum Sex: String, CaseIterable {
case male, female
case undisclosed = ""
var label: String {
switch self {
case .male: return ""
case .female: return ""
case .undisclosed: return "不愿透露"
}
}
}
var sex: Sex {
get { Sex(rawValue: biologicalSexRaw) ?? .undisclosed }
set { biologicalSexRaw = newValue.rawValue }
}
/// birthYear nil,
var age: Int? {
guard let y = birthYear else { return nil }
return Calendar.current.component(.year, from: .now) - y
}
/// ProfileCard :"38 · · 175cm · 68kg · A"
var summaryLine: String {
var parts: [String] = []
if let age { parts.append("\(age)") }
if sex != .undisclosed { parts.append(sex.label) }
if let h = heightCM { parts.append("\(h)cm") }
if let w = weightKG {
let s = w.truncatingRemainder(dividingBy: 1) == 0
? String(format: "%.0fkg", w)
: String(format: "%.1fkg", w)
parts.append(s)
}
if !bloodTypeRaw.isEmpty { parts.append("\(bloodTypeRaw)") }
return parts.joined(separator: " · ")
}
/// summaryLine("")
var hasAnyBasics: Bool {
birthYear != nil ||
sex != .undisclosed ||
heightCM != nil ||
weightKG != nil ||
!bloodTypeRaw.isEmpty
}
/// BMI(kg/m²), +
var bmi: Double? {
guard let h = heightCM, h > 0, let w = weightKG else { return nil }
let m = Double(h) / 100.0
return w / (m * m)
}
}
/// : App UserProfile
enum UserProfileStore {
@MainActor
static func loadOrCreate(in ctx: ModelContext) -> UserProfile {
let descriptor = FetchDescriptor<UserProfile>()
if let existing = try? ctx.fetch(descriptor).first {
return existing
}
let new = UserProfile()
ctx.insert(new)
try? ctx.save()
return new
}
}