feat(ios): 添加用药计划Widget小组件支持

- 创建medicineExtension小组件,支持iOS桌面显示用药计划
- 实现App Group数据共享机制,支持主应用与小组件数据同步
- 添加AppGroupUserDefaultsManager原生模块,提供跨应用数据访问能力
- 添加WidgetManager和WidgetCenterHelper,实现小组件刷新控制
- 在medications页面和Redux store中集成小组件数据同步逻辑
- 支持实时同步今日用药状态(待服用/已服用/已错过)到小组件
- 配置App Group entitlements (group.com.anonymous.digitalpilates)
- 更新Xcode项目配置,添加WidgetKit和SwiftUI框架支持
This commit is contained in:
richarjiang
2025-11-14 08:51:02 +08:00
parent d282abd146
commit b0e93eedae
25 changed files with 1423 additions and 4 deletions

View File

@@ -0,0 +1,80 @@
//
// medicineLiveActivity.swift
// medicine
//
// Created by richard on 2025/11/13.
//
import ActivityKit
import WidgetKit
import SwiftUI
struct medicineAttributes: ActivityAttributes {
public struct ContentState: Codable, Hashable {
// Dynamic stateful properties about your activity go here!
var emoji: String
}
// Fixed non-changing properties about your activity go here!
var name: String
}
struct medicineLiveActivity: Widget {
var body: some WidgetConfiguration {
ActivityConfiguration(for: medicineAttributes.self) { context in
// Lock screen/banner UI goes here
VStack {
Text("Hello \(context.state.emoji)")
}
.activityBackgroundTint(Color.cyan)
.activitySystemActionForegroundColor(Color.black)
} dynamicIsland: { context in
DynamicIsland {
// Expanded UI goes here. Compose the expanded UI through
// various regions, like leading/trailing/center/bottom
DynamicIslandExpandedRegion(.leading) {
Text("Leading")
}
DynamicIslandExpandedRegion(.trailing) {
Text("Trailing")
}
DynamicIslandExpandedRegion(.bottom) {
Text("Bottom \(context.state.emoji)")
// more content
}
} compactLeading: {
Text("L")
} compactTrailing: {
Text("T \(context.state.emoji)")
} minimal: {
Text(context.state.emoji)
}
.widgetURL(URL(string: "http://www.apple.com"))
.keylineTint(Color.red)
}
}
}
extension medicineAttributes {
fileprivate static var preview: medicineAttributes {
medicineAttributes(name: "World")
}
}
extension medicineAttributes.ContentState {
fileprivate static var smiley: medicineAttributes.ContentState {
medicineAttributes.ContentState(emoji: "😀")
}
fileprivate static var starEyes: medicineAttributes.ContentState {
medicineAttributes.ContentState(emoji: "🤩")
}
}
#Preview("Notification", as: .content, using: medicineAttributes.preview) {
medicineLiveActivity()
} contentStates: {
medicineAttributes.ContentState.smiley
medicineAttributes.ContentState.starEyes
}