Files
digital-pilates/ios/medicine/medicine.swift
richarjiang 7bd0b5fc52 # 方案总结
基于提供的 Git diff,我将生成以下 conventional commit message:

## 变更分析:

1. **核心功能**:
   - 新增睡眠监控服务(`services/sleepMonitor.ts`)
   - 新增睡眠通知服务(`services/sleepNotificationService.ts`)
   - iOS 原生端增加睡眠观察者方法

2. **应用启动优化**:
   - 重构 `app/_layout.tsx` 中的初始化流程,按优先级分阶段加载服务

3. **药品功能改进**:
   - 优化语音识别交互(实时预览、可取消)
   - Widget 增加 URL scheme 支持

4. **路由配置**:
   - 新增药品管理路由常量

## 提交信息类型:
- **主类型**:`feat` (新增睡眠监控功能)
- **作用域**:`health` (健康相关功能)

---

请确认方案后,我将生成最终的 commit message。

---

**最终 Commit Message:**

feat(health): 添加睡眠监控和通知服务,优化应用启动流程

- 新增睡眠监控服务,支持实时监听 HealthKit 睡眠数据更新
- 实现睡眠质量分析算法,计算睡眠评分和各阶段占比
- 新增睡眠通知服务,分析完成后自动推送质量评估和建议
- iOS 原生端实现睡眠数据观察者,支持后台数据传递
- 重构应用启动初始化流程,按优先级分阶段加载服务(关键/次要/后台/空闲)
- 优化药品录入页面语音识别交互,支持实时预览和取消操作
- 药品 Widget 增加 deeplink 支持,点击跳转到应用
- 新增药品管理路由常量配置
2025-11-14 10:52:26 +08:00

298 lines
9.1 KiB
Swift
Raw Permalink Blame History

This file contains ambiguous Unicode characters

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

//
// medicine.swift
// medicine
//
// Created by richard on 2025/11/13.
//
import WidgetKit
import SwiftUI
// MARK: -
struct MedicationItem: Codable, Identifiable {
let id: String
let name: String
let dosage: String
let scheduledTime: String
let status: String
let medicationId: String
let recordId: String?
let image: String?
}
struct MedicationWidgetData: Codable {
let medications: [MedicationItem]
let lastSyncTime: String
let date: String
}
// MARK: - Timeline Provider
struct Provider: AppIntentTimelineProvider {
func placeholder(in context: Context) -> MedicationEntry {
MedicationEntry(
date: Date(),
medications: [],
displayDate: formatDate(Date()),
lastSyncTime: Date().timeIntervalSince1970
)
}
func snapshot(for configuration: ConfigurationAppIntent, in context: Context) async -> MedicationEntry {
let medicationData = loadMedicationDataFromAppGroup()
let lastSyncTimeInterval = parseTimeInterval(from: medicationData.lastSyncTime)
return MedicationEntry(
date: Date(),
medications: medicationData.medications,
displayDate: medicationData.date,
lastSyncTime: lastSyncTimeInterval
)
}
func timeline(for configuration: ConfigurationAppIntent, in context: Context) async -> Timeline<MedicationEntry> {
let currentDate = Date()
let medicationData = loadMedicationDataFromAppGroup()
let lastSyncTimeInterval = parseTimeInterval(from: medicationData.lastSyncTime)
let entry = MedicationEntry(
date: currentDate,
medications: medicationData.medications,
displayDate: medicationData.date,
lastSyncTime: lastSyncTimeInterval
)
// - 15
let nextUpdate = Calendar.current.date(byAdding: .minute, value: 15, to: currentDate)!
return Timeline(entries: [entry], policy: .after(nextUpdate))
}
}
// MARK: - Timeline Entry
struct MedicationEntry: TimelineEntry {
let date: Date
let medications: [MedicationItem]
let displayDate: String
let lastSyncTime: TimeInterval
}
// MARK: - Widget View
struct medicineEntryView: View {
var entry: Provider.Entry
var body: some View {
VStack(spacing: 8) {
//
Text(entry.displayDate)
.font(.caption)
.foregroundColor(.secondary)
//
if entry.medications.isEmpty {
Text("今日无用药计划")
.font(.body)
.foregroundColor(.secondary)
} else {
LazyVStack(spacing: 6) {
ForEach(entry.medications, id: \.id) { medication in
MedicationRowView(medication: medication)
}
}
}
Spacer()
//
Text("最后更新: \(formatTime(from: entry.lastSyncTime))")
.font(.caption2)
.foregroundColor(Color.secondary)
}
.padding()
.widgetURL(URL(string: "digitalpilates://medications"))
}
}
// MARK: -
struct MedicationRowView: View {
let medication: MedicationItem
var body: some View {
HStack {
//
Circle()
.fill(statusColor)
.frame(width: 8, height: 8)
VStack(alignment: .leading, spacing: 2) {
Text(medication.name)
.font(.caption)
.fontWeight(.medium)
.lineLimit(1)
Text("\(medication.scheduledTime)\(medication.dosage)")
.font(.caption2)
.foregroundColor(.secondary)
}
Spacer()
//
Text(statusText)
.font(.caption2)
.padding(.horizontal, 8)
.padding(.vertical, 2)
.background(statusColor.opacity(0.2))
.foregroundColor(statusColor)
.cornerRadius(4)
}
.padding(.vertical, 2)
}
private var statusColor: Color {
switch medication.status {
case "taken":
return .green
case "missed":
return .red
default:
return .blue
}
}
private var statusText: String {
switch medication.status {
case "taken":
return "已服用"
case "missed":
return "已错过"
default:
return "待服用"
}
}
}
// MARK: - Widget Configuration
struct medicine: Widget {
let kind: String = "medicine"
var body: some WidgetConfiguration {
AppIntentConfiguration(kind: kind, intent: ConfigurationAppIntent.self, provider: Provider()) { entry in
medicineEntryView(entry: entry)
.containerBackground(.white, for: .widget)
}
.configurationDisplayName("用药计划")
.description("显示今日用药计划和服药状态")
.supportedFamilies([.systemSmall, .systemMedium])
}
}
// MARK: - Helper Functions
private func loadMedicationDataFromAppGroup() -> MedicationWidgetData {
guard let userDefaults = UserDefaults(suiteName: "group.com.anonymous.digitalpilates") else {
print("❌ Failed to initialize UserDefaults with App Group")
return MedicationWidgetData(
medications: [],
lastSyncTime: Date().toISOString(),
date: formatDate(Date())
)
}
guard let dataJson = userDefaults.string(forKey: "widget_medication_data") else {
print("⚠️ No medication data found in App Group UserDefaults")
return MedicationWidgetData(
medications: [],
lastSyncTime: Date().toISOString(),
date: formatDate(Date())
)
}
print("✅ Found medication data JSON: \(dataJson.prefix(100))...")
guard let data = dataJson.data(using: .utf8) else {
print("❌ Failed to convert JSON string to Data")
return MedicationWidgetData(
medications: [],
lastSyncTime: Date().toISOString(),
date: formatDate(Date())
)
}
do {
let medicationData = try JSONDecoder().decode(MedicationWidgetData.self, from: data)
print("✅ Successfully decoded medication data: \(medicationData.medications.count) medications")
return medicationData
} catch {
print("❌ Failed to decode medication data: \(error)")
return MedicationWidgetData(
medications: [],
lastSyncTime: Date().toISOString(),
date: formatDate(Date())
)
}
}
private func formatDate(_ date: Date) -> String {
let formatter = DateFormatter()
formatter.dateFormat = "M月d日"
return formatter.string(from: date)
}
private func formatTime(from timeInterval: TimeInterval) -> String {
let date = Date(timeIntervalSince1970: timeInterval)
let formatter = DateFormatter()
formatter.timeStyle = .short
formatter.locale = Locale(identifier: "zh_CN")
return formatter.string(from: date)
}
/// - ISO
private func parseTimeInterval(from string: String) -> TimeInterval {
//
if let timestamp = Double(string) {
return timestamp
}
// ISO 8601
let isoFormatter = ISO8601DateFormatter()
isoFormatter.formatOptions = [.withInternetDateTime, .withFractionalSeconds]
if let date = isoFormatter.date(from: string) {
return date.timeIntervalSince1970
}
// ISO
isoFormatter.formatOptions = [.withInternetDateTime]
if let date = isoFormatter.date(from: string) {
return date.timeIntervalSince1970
}
//
return Date().timeIntervalSince1970
}
/// DateISO
extension Date {
func toISOString() -> String {
let formatter = ISO8601DateFormatter()
formatter.formatOptions = [.withInternetDateTime, .withFractionalSeconds]
return formatter.string(from: self)
}
}
extension ConfigurationAppIntent {
fileprivate static var smiley: ConfigurationAppIntent {
let intent = ConfigurationAppIntent()
intent.favoriteEmoji = "😀"
return intent
}
fileprivate static var starEyes: ConfigurationAppIntent {
let intent = ConfigurationAppIntent()
intent.favoriteEmoji = "🤩"
return intent
}
}
#Preview(as: .systemSmall) {
medicine()
} timeline: {
MedicationEntry(date: .now, medications: [], displayDate: formatDate(Date()), lastSyncTime: Date().timeIntervalSince1970)
MedicationEntry(date: .now, medications: [], displayDate: formatDate(Date()), lastSyncTime: Date().timeIntervalSince1970)
}