Files
kangkang/康康/Features/Timeline/DateSection.swift
link2026 d2c77d5c51 feat: 国际化(i18n) en/ja/ko + App 内语言切换
主体:多语言支持(简体中文源 + 英/日/韩)
- 基础设施:Localizable.xcstrings(String Catalog,sourceLanguage=zh-Hans)
  + pbxproj developmentRegion/knownRegions 注册 en/ja/ko
- 全部硬编码 Locale("zh_CN") → Locale.current;中文 dateFormat → Date.FormatStyle(跟随系统)
- UI 中文字面量统一为 String(appLoc:)(显式绑定所选语言 bundle+locale,即时切换)
  Text 字面量走环境 \.locale + Bundle 重定向
- 549 个 catalog key 全部 en/ja/ko 翻译完成(0 未翻译)
- App 内语言切换:我的 → 语言(LanguageManager + 即时生效,无需重启)
- 双用预设(症状/监测指标/慢病)本地化:static→computed 避免缓存

注:本提交为 WIP,一并打包了并行进行的功能模块
(HealthExport 健康导出、Security/Face ID 锁、DiaryAssist 日记 AI 辅助)
及 App 图标、CLAUDE.md、docs/scripts。

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-05-30 10:28:24 +08:00

78 lines
2.7 KiB
Swift

import Foundation
nonisolated enum DateSection: Hashable {
case today
case yesterday
case thisWeek
case thisMonth
case year(Int)
var label: String {
switch self {
case .today: return String(appLoc: "今天")
case .yesterday: return String(appLoc: "昨天")
case .thisWeek: return String(appLoc: "本周")
case .thisMonth: return String(appLoc: "本月")
case .year(let y): return String(appLoc: "\(y)")
}
}
var sortIndex: Int {
switch self {
case .today: return 0
case .yesterday: return 1
case .thisWeek: return 2
case .thisMonth: return 3
case .year(let y): return 10_000 - y
}
}
}
enum TimelineGrouping {
static func section(for date: Date,
now: Date = .now,
calendar: Calendar = .current) -> DateSection {
if calendar.isDate(date, inSameDayAs: now) { return .today }
if let yesterday = calendar.date(byAdding: .day, value: -1, to: now),
calendar.isDate(date, inSameDayAs: yesterday) {
return .yesterday
}
if calendar.isDate(date, equalTo: now, toGranularity: .weekOfYear) {
return .thisWeek
}
if calendar.isDate(date, equalTo: now, toGranularity: .month) {
return .thisMonth
}
let year = calendar.component(.year, from: date)
return .year(year)
}
static func group(_ entries: [TimelineEntry],
now: Date = .now,
calendar: Calendar = .current)
-> [(section: DateSection, items: [TimelineEntry])] {
var buckets: [DateSection: [TimelineEntry]] = [:]
for entry in entries {
let key = section(for: entry.date, now: now, calendar: calendar)
buckets[key, default: []].append(entry)
}
return buckets
.map { ($0.key, $0.value.sorted { $0.date > $1.date }) }
.sorted { $0.0.sortIndex < $1.0.sortIndex }
}
}
func formatDuration(_ interval: TimeInterval) -> String {
let totalMinutes = Int(max(0, interval) / 60)
let days = totalMinutes / (60 * 24)
let hours = (totalMinutes % (60 * 24)) / 60
let minutes = totalMinutes % 60
if days > 0 && hours > 0 { return String(appLoc: "\(days)\(hours) 小时") }
if days > 0 { return String(appLoc: "\(days)") }
if hours > 0 && minutes > 0 { return String(appLoc: "\(hours) 小时 \(minutes)") }
if hours > 0 { return String(appLoc: "\(hours) 小时") }
if minutes > 0 { return String(appLoc: "\(minutes) 分钟") }
return String(appLoc: "刚刚")
}