Files
kangkang/docs/superpowers/specs/2026-05-30-custom-reminder-design.md
link2026 7ad41c5f09 ```
docs(health-profile): 添加防编造加固修订记录到导出健康档案设计文档

补充了关于导出摘要出现虚构病例问题的详细分析和修复方案,
包括检索策略优化、空数据兜底处理和prompt重写等三层防护措施。
```
2026-05-30 20:06:12 +08:00

7.6 KiB
Raw Permalink Blame History

自由周期提醒(CustomReminder)— 设计文档

日期:2026-05-30(W2) 作者:link2026 + Claude 关联卖点:#4 隐私三件套之外的实用粘性功能(本地通知,无云) 优先级:用户明确要求(注:§10.6「用药提醒」原列默认不做,本轮经讨论确认要做,按最小可用实现)


1. 一句话定位

让用户新建自由文案的周期性本地提醒(如「每天 20:00 跑步 5 公里」「每天 12:30 吃 2 片护肝片」),与现有「指标记录提醒」(去录某项指标)并存但相互独立。完全本地 UserNotifications,不引云。


2. 已确认的设计决策

决策点 选择
模型 新建独立 CustomReminder @Model,不动现有 MetricReminder
周期粒度 每日 / 每周选几天 / 每月某日 / 每年某月某日(2026-05-30 用户反转原「不做按月/按年」决策)。仍不做「每 N 天间隔」/一次性
时间选择 常用时间快捷预设(8:00/12:00/18:00/22:00 chip)+ 保留 DatePicker 精调
入口 新建 → 开启一个提醒 → RemindersListView(提醒中心),顶部「+ 新建提醒」打开编辑 sheet
列表范围 自由提醒 + 指标提醒合展(上次删了「我的」入口,指标提醒也只能从这里管)
量词(5公里/2片) 写在自由文本 title 里,不单设字段
多语言 所有固定文案走 String(appLoc:),新增中文 key 补 en/ja/ko 到 Localizable.xcstrings

3. 数据模型

Models/Models.swift 新增:

@Model final class CustomReminder {
    enum Frequency: String { case daily, weekly, monthly, yearly }   // 嵌套枚举
    @Attribute(.unique) var id: UUID
    var title: String                    // 用户文案:"跑步5公里"
    var note: String                     // 可选备注 → 通知正文
    var hour: Int                        // 0...23
    var minute: Int                      // 0...59
    var weekdays: [Int]                  // 1=日…7=六,仅 weekly 用(复用 MetricReminder 约定)
    var frequencyRaw: String = "daily"   // Frequency 原始值(内联默认 → 走轻量迁移)
    var dayOfMonth: Int = 1              // monthly / yearly 用,1...31
    var month: Int = 1                   // yearly 用,1...12
    var enabled: Bool
    var createdAt: Date
    var updatedAt: Date
    // computed: frequency(get/set 包 frequencyRaw)/ isEveryDay / frequencyLabel(分档)/ timeLabel
}

Schema 已含 CustomReminder.self本轮只给已存在的 CustomReminder 加 3 个带内联默认值的属性 → SwiftData 自动轻量迁移,不触发删库兜底(见 §10)。

四档语义 → iOS UNCalendarNotificationTrigger(repeats:true):

频率 DateComponents 通知数 id 后缀
daily hour,minute 1 .daily
weekly hour,minute,weekday ×N N .w<weekday>
monthly day,hour,minute 1 .monthly
yearly month,day,hour,minute 1 .yearly

边界:iOS 重复触发不顺延。monthly 选 29/30/31 → 无此日的月份跳过(UI 给浅色提示);yearly 的「日」选项按所选月份最大天数动态收口(避免「4月31日」永不触发),仅闰年 2/29 给提示。


4. 通知调度(ReminderService 泛化)

抽出私有共享核心,两种提醒复用:

private static func schedule(idBase:title:body:hour:minute:weekdays:thread:) async
static func sync(_ custom: CustomReminder) async    // 新增
static func cancel(customId: UUID)                  // 新增
static func sync(_ metric: MetricReminder) async    // 现有,内部改走共享核心,行为不变
  • custom 通知:title = 提醒标题,body = 备注(空则用默认文案「到点啦,记得完成」)。
  • id 前缀 kangkang.custom.<uuid>.w<weekday>(与指标的 kangkang.reminder.<metricId>.w<weekday> 不冲突)。
  • 保存时调 requestAuthorization();被拒则提示去系统设置。

5. UI

5.1 CustomReminderEditSheet(新增)

创建 / 编辑共用。字段:

  • 标题 TextField(占位:「做点什么?例:跑步5公里 / 吃2片护肝片」),空标题禁用保存。
  • 备注 TextField(可选)。
  • 时间 DatePicker(.hourAndMinute)。
  • 周几选择(复用 RemindersListView 的 chip 行)。
  • 保存 / 取消;编辑态多一个「删除提醒」。 保存:写 SwiftData → 请求通知权限 → ReminderService.sync(custom)

5.2 RemindersListView(改造为提醒中心)

  • 顶部「+ 新建提醒」按钮 → 打开 CustomReminderEditSheet(create)。
  • 「我的提醒」区:@Query CustomReminder,每行点开走编辑 sheet,行上 Toggle 控 enabled。
  • 「指标记录提醒」区:@Query MetricReminder,保持现有内联编辑不变(仅非空时显示区头)。
  • 表头副文案、空状态文案更新。

6. 多语言

新增中文 key + en/ja/ko 译文写入 Localizable.xcstrings(源语言 zh-Hans,key 即中文)。脚本只增不改,已存在的 key 跳过。复用已有 key:时间/保存/取消/删除提醒/每天/已关闭/周几名等。用户输入的标题/备注是数据,不翻译。


7. 文件清单

文件 改动
Models/Models.swift CustomReminder +Frequency 枚举 +frequencyRaw/dayOfMonth/month(均带内联默认)+ 分档 frequencyLabel
App/KangkangApp.swift 持久化兜底改造:迁移失败时由「删库」改为「挪到 StoreBackups/<时间戳>/ 再重建」(见 §10)
Services/ReminderService.swift 调度核心泛化为 Slot(suffix,DateComponents) 列表;custom sync 按 frequency 分档;cancelBase 覆盖 daily/monthly/yearly/w1-7
Features/Me/CustomReminderEditSheet.swift 频率分段 Picker + 各档子控件(周几 / 日 / 月+日)+ 时间快捷预设行
Features/Me/RemindersListView.swift 不变(frequencyLabel 来自模型)
Localizable.xcstrings 新增 11 个 key × en/ja/ko

8. 红线对齐

  • 不引云、不碰密码学(纯本地通知)
  • 不重构 Tab/RecordSheet 骨架
  • §10.6「用药提醒默认不做」→ 已讨论确认,最小实现(无贪睡/铃声/间隔)

9. 验收(真机)

① 新建「每天 20:00 跑步 5 公里」→ 列表出现 → 到点收到本地通知(标题=跑步5公里);② 改时间/周几即时重排;③ 关闭 Toggle 取消通知;④ 删除清除 pending;⑤ 切换语言后固定文案随之变化(用户输入文案不变);⑥ 指标提醒仍在同一列表可管;⑦ 每月/每年:切频率后子控件随之变化,边界提示出现;改频率后旧档 pending 通知被清掉(不留孤儿);⑧ 时间预设:点 8:00/12:00/18:00/22:00 即填,精调仍可用。


10. 顺带修复:重打包数据丢失(根因 + 方案)

问题:Demo 期每次改 schema 重打包,SwiftData 数据被清空。

根因(单点):App/KangkangApp.swiftModelContainer 创建 catch 块直接删 store 文件。SwiftData 只对纯增量改动自动轻量迁移;一旦某次改动超纲(最常见:给已存在的 @Model 新增「非可选且无内联默认值」的属性),自动迁移抛错 → 落入 catch → 删库。W2 几乎每次都在改 schema,故体感「每次都丢」。

方案(两层):

  1. 治本:新增 @Model 属性一律「可选」或「内联默认值」(本轮 3 个新字段都给了 = "daily" / = 1)→ 走轻量迁移、不进 catch、数据保留。
  2. 兜底:catch 不再删库,改为把旧 store(含 -wal/-shm)挪到 Application Support/StoreBackups/<时间戳>/ 再重建——App 仍能启动,旧数据可手动恢复;挪不动才降级删除。

⚠️ 正式发布前仍应升级为 VersionedSchema + SchemaMigrationPlan 的正式迁移(注释已就地标注)。