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。
This commit is contained in:
link2026
2026-05-26 07:58:47 +08:00
parent 37b47b2076
commit 39edc25dc1
5 changed files with 83 additions and 31 deletions

View File

@@ -601,14 +601,6 @@ struct IndicatorQuickSheet: View {
) )
ctx.insert(indicator) ctx.insert(indicator)
try? ctx.save() try? ctx.save()
// Profile
if m == .height, let cm = Int(value.trimmingCharacters(in: .whitespaces)),
let p = profile {
p.heightCM = cm
p.updatedAt = .now
try? ctx.save()
}
} }
private func saveFreeform() { private func saveFreeform() {

View File

@@ -9,11 +9,11 @@ enum MonitorMetric: String, CaseIterable, Identifiable {
case bloodPressure // bp.systolic + bp.diastolic case bloodPressure // bp.systolic + bp.diastolic
case fastingGlucose // glucose.fasting case fastingGlucose // glucose.fasting
case postprandialGlucose // glucose.postprandial case postprandialGlucose // glucose.postprandial
case weight // weight
case temperature // temperature case temperature // temperature
case heartRate // heart_rate case heartRate // heart_rate
case spo2 // spo2 case spo2 // spo2
case height // height( UserProfile.heightCM)
// : / UserProfile (,)
var id: String { rawValue } var id: String { rawValue }
@@ -22,11 +22,9 @@ enum MonitorMetric: String, CaseIterable, Identifiable {
case .bloodPressure: return "血压" case .bloodPressure: return "血压"
case .fastingGlucose: return "空腹血糖" case .fastingGlucose: return "空腹血糖"
case .postprandialGlucose: return "餐后血糖" case .postprandialGlucose: return "餐后血糖"
case .weight: return "体重"
case .temperature: return "体温" case .temperature: return "体温"
case .heartRate: return "心率" case .heartRate: return "心率"
case .spo2: return "血氧" case .spo2: return "血氧"
case .height: return "身高"
} }
} }
@@ -36,11 +34,9 @@ enum MonitorMetric: String, CaseIterable, Identifiable {
case .bloodPressure: return "heart.fill" case .bloodPressure: return "heart.fill"
case .fastingGlucose: return "drop.fill" case .fastingGlucose: return "drop.fill"
case .postprandialGlucose: return "drop.circle.fill" case .postprandialGlucose: return "drop.circle.fill"
case .weight: return "scalemass.fill"
case .temperature: return "thermometer.medium" case .temperature: return "thermometer.medium"
case .heartRate: return "waveform.path.ecg" case .heartRate: return "waveform.path.ecg"
case .spo2: return "lungs.fill" case .spo2: return "lungs.fill"
case .height: return "ruler.fill"
} }
} }
@@ -71,12 +67,6 @@ enum MonitorMetric: String, CaseIterable, Identifiable {
unit: "mmol/L", unit: "mmol/L",
placeholder: "6.5", placeholder: "6.5",
baseRange: 0...7.8)] baseRange: 0...7.8)]
case .weight:
return [Field(seriesKey: "weight",
label: "体重",
unit: "kg",
placeholder: "68",
baseRange: nil)]
case .temperature: case .temperature:
return [Field(seriesKey: "temperature", return [Field(seriesKey: "temperature",
label: "体温", label: "体温",
@@ -95,12 +85,6 @@ enum MonitorMetric: String, CaseIterable, Identifiable {
unit: "%", unit: "%",
placeholder: "98", placeholder: "98",
baseRange: 95...100)] baseRange: 95...100)]
case .height:
return [Field(seriesKey: "height",
label: "身高",
unit: "cm",
placeholder: "175",
baseRange: nil)]
} }
} }
} }

View File

@@ -53,11 +53,28 @@ private struct ProfileEditForm: View {
// MARK: - // MARK: -
private var basicsSection: some View { private var basicsSection: some View {
Section("基本") { Section {
birthYearPicker birthYearPicker
sexPicker sexPicker
heightRow heightRow
weightRow
bloodTypePicker bloodTypePicker
} header: {
Text("基本")
} footer: {
if let bmi = profile.bmi {
Text("BMI: \(String(format: "%.1f", bmi)) \(bmiLabel(bmi))")
.font(.system(size: 11))
}
}
}
private func bmiLabel(_ bmi: Double) -> String {
switch bmi {
case ..<18.5: return "(偏瘦)"
case ..<24: return "(正常)"
case ..<28: return "(超重)"
default: return "(肥胖)"
} }
} }
@@ -97,6 +114,18 @@ private struct ProfileEditForm: View {
} }
} }
private var weightRow: some View {
HStack {
Text("体重")
Spacer()
TextField("kg", value: $profile.weightKG, format: .number.precision(.fractionLength(0...1)))
.keyboardType(.decimalPad)
.multilineTextAlignment(.trailing)
.frame(width: 80)
Text("kg").foregroundStyle(Tj.Palette.text3)
}
}
private var bloodTypePicker: some View { private var bloodTypePicker: some View {
Picker("血型", selection: $profile.bloodTypeRaw) { Picker("血型", selection: $profile.bloodTypeRaw) {
Text("不知道").tag("") Text("不知道").tag("")

View File

@@ -3,10 +3,11 @@ import SwiftData
@Model @Model
final class UserProfile { final class UserProfile {
// 4 // 5
var birthYear: Int? // 1990, var birthYear: Int? // 1990,
var biologicalSexRaw: String // "" / "male" / "female" var biologicalSexRaw: String // "" / "male" / "female"
var heightCM: Int? var heightCM: Int?
var weightKG: Double? // (68.5)
var bloodTypeRaw: String // "" / "A" / "B" / "AB" / "O" var bloodTypeRaw: String // "" / "A" / "B" / "AB" / "O"
// //
@@ -22,6 +23,7 @@ final class UserProfile {
init(birthYear: Int? = nil, init(birthYear: Int? = nil,
biologicalSexRaw: String = "", biologicalSexRaw: String = "",
heightCM: Int? = nil, heightCM: Int? = nil,
weightKG: Double? = nil,
bloodTypeRaw: String = "", bloodTypeRaw: String = "",
allergies: [String] = [], allergies: [String] = [],
chronicConditions: [String] = [], chronicConditions: [String] = [],
@@ -31,6 +33,7 @@ final class UserProfile {
self.birthYear = birthYear self.birthYear = birthYear
self.biologicalSexRaw = biologicalSexRaw self.biologicalSexRaw = biologicalSexRaw
self.heightCM = heightCM self.heightCM = heightCM
self.weightKG = weightKG
self.bloodTypeRaw = bloodTypeRaw self.bloodTypeRaw = bloodTypeRaw
self.allergies = allergies self.allergies = allergies
self.chronicConditions = chronicConditions self.chronicConditions = chronicConditions
@@ -65,19 +68,36 @@ extension UserProfile {
return Calendar.current.component(.year, from: .now) - y return Calendar.current.component(.year, from: .now) - y
} }
/// ProfileCard :"38 · · 175cm · A" /// ProfileCard :"38 · · 175cm · 68kg · A"
var summaryLine: String { var summaryLine: String {
var parts: [String] = [] var parts: [String] = []
if let age { parts.append("\(age)") } if let age { parts.append("\(age)") }
if sex != .undisclosed { parts.append(sex.label) } if sex != .undisclosed { parts.append(sex.label) }
if let h = heightCM { parts.append("\(h)cm") } 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)") } if !bloodTypeRaw.isEmpty { parts.append("\(bloodTypeRaw)") }
return parts.joined(separator: " · ") return parts.joined(separator: " · ")
} }
/// summaryLine("") /// summaryLine("")
var hasAnyBasics: Bool { var hasAnyBasics: Bool {
birthYear != nil || sex != .undisclosed || heightCM != nil || !bloodTypeRaw.isEmpty 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)
} }
} }

View File

@@ -17,6 +17,7 @@ struct UserProfileTests {
let p = UserProfile() let p = UserProfile()
#expect(p.birthYear == nil) #expect(p.birthYear == nil)
#expect(p.heightCM == nil) #expect(p.heightCM == nil)
#expect(p.weightKG == nil)
#expect(p.biologicalSexRaw == "") #expect(p.biologicalSexRaw == "")
#expect(p.bloodTypeRaw == "") #expect(p.bloodTypeRaw == "")
#expect(p.allergies.isEmpty) #expect(p.allergies.isEmpty)
@@ -25,6 +26,7 @@ struct UserProfileTests {
#expect(p.familyHistory.isEmpty) #expect(p.familyHistory.isEmpty)
#expect(p.age == nil) #expect(p.age == nil)
#expect(p.sex == .undisclosed) #expect(p.sex == .undisclosed)
#expect(p.bmi == nil)
#expect(p.hasAnyBasics == false) #expect(p.hasAnyBasics == false)
#expect(p.summaryLine == "") #expect(p.summaryLine == "")
} }
@@ -50,9 +52,34 @@ struct UserProfileTests {
#expect(line.contains("")) #expect(line.contains(""))
#expect(line.contains("")) #expect(line.contains(""))
#expect(line.contains("162cm")) #expect(line.contains("162cm"))
#expect(!line.contains("kg")) // ,
#expect(!line.contains("")) // , #expect(!line.contains("")) // ,
} }
@Test func summaryLineIncludesWeightWhenSet() {
let p = UserProfile(heightCM: 175, weightKG: 68.5)
#expect(p.summaryLine.contains("175cm"))
#expect(p.summaryLine.contains("68.5kg"))
}
@Test func summaryLineFormatsIntegerWeightWithoutDecimal() {
let p = UserProfile(weightKG: 70)
#expect(p.summaryLine == "70kg")
}
@Test func bmiNilWithoutBothHeightAndWeight() {
#expect(UserProfile(heightCM: 175).bmi == nil)
#expect(UserProfile(weightKG: 68).bmi == nil)
#expect(UserProfile().bmi == nil)
}
@Test func bmiComputedFromHeightAndWeight() throws {
let p = UserProfile(heightCM: 175, weightKG: 70)
let bmi = try #require(p.bmi)
// 70 / (1.75 * 1.75) = 22.857
#expect(abs(bmi - 22.857) < 0.01)
}
@Test func loadOrCreateReturnsExistingSingleton() throws { @Test func loadOrCreateReturnsExistingSingleton() throws {
let ctx = try makeContext() let ctx = try makeContext()
let first = UserProfileStore.loadOrCreate(in: ctx) let first = UserProfileStore.loadOrCreate(in: ctx)