``` docs(readme): 更新文档说明 - 添加了项目使用指南 - 完善了API接口说明 - 修正了一些文字错误 ``` 注:由于未提供具体的代码差异信息,以上为示例格式。请提供具体的代码变更内容以便生成准确的commit message。
144 lines
4.9 KiB
Swift
144 lines
4.9 KiB
Swift
import SwiftUI
|
|
import SwiftData
|
|
|
|
/// 「我的导出」全部历史列表。从 ArchiveListView 顶部 strip 的「查看全部」进入。
|
|
struct HealthExportListView: View {
|
|
@Environment(\.modelContext) private var ctx
|
|
@Query(sort: \HealthExport.createdAt, order: .reverse)
|
|
private var exports: [HealthExport]
|
|
|
|
@State private var selected: HealthExport?
|
|
|
|
var body: some View {
|
|
VStack(alignment: .leading, spacing: 0) {
|
|
header
|
|
.padding(.horizontal, 20)
|
|
.padding(.top, 8)
|
|
.padding(.bottom, 14)
|
|
|
|
if exports.isEmpty {
|
|
empty
|
|
} else {
|
|
ScrollView {
|
|
LazyVStack(spacing: 12) {
|
|
ForEach(exports) { exp in
|
|
Button {
|
|
selected = exp
|
|
} label: {
|
|
HealthExportRow(export: exp)
|
|
}
|
|
.buttonStyle(.plain)
|
|
.contextMenu {
|
|
Button(role: .destructive) {
|
|
delete(exp)
|
|
} label: {
|
|
Label("删除", systemImage: "trash")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.padding(.horizontal, 20)
|
|
.padding(.bottom, 24)
|
|
}
|
|
}
|
|
}
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .topLeading)
|
|
.background(Tj.Palette.sand.ignoresSafeArea())
|
|
.navigationTitle("我的导出")
|
|
.navigationBarTitleDisplayMode(.inline)
|
|
.sheet(item: $selected) { exp in
|
|
HealthExportDetailView(export: exp)
|
|
}
|
|
}
|
|
|
|
private var header: some View {
|
|
HStack(alignment: .lastTextBaseline) {
|
|
Text("我的导出")
|
|
.font(.tjTitle(24))
|
|
.foregroundStyle(Tj.Palette.text)
|
|
Text(exports.isEmpty ? "" : String(appLoc: "\(exports.count) 份"))
|
|
.font(.tjScaled( 12))
|
|
.foregroundStyle(Tj.Palette.text3)
|
|
Spacer()
|
|
TjLockChip()
|
|
}
|
|
}
|
|
|
|
private var empty: some View {
|
|
VStack(spacing: 12) {
|
|
Spacer()
|
|
TjPlaceholder(label: String(appLoc: "还没有导出过\n回到记录页右上角生成一份"))
|
|
.frame(width: 240, height: 140)
|
|
Spacer()
|
|
}
|
|
.frame(maxWidth: .infinity)
|
|
}
|
|
|
|
private func delete(_ exp: HealthExport) {
|
|
ctx.delete(exp)
|
|
try? ctx.save()
|
|
}
|
|
}
|
|
|
|
/// 列表里一条行。
|
|
struct HealthExportRow: View {
|
|
let export: HealthExport
|
|
|
|
var body: some View {
|
|
VStack(alignment: .leading, spacing: 6) {
|
|
HStack(alignment: .top) {
|
|
Text(export.promptPreview)
|
|
.font(.tjScaled( 14, weight: .semibold))
|
|
.foregroundStyle(Tj.Palette.text)
|
|
.lineLimit(2)
|
|
.multilineTextAlignment(.leading)
|
|
Spacer()
|
|
Image(systemName: "chevron.right")
|
|
.font(.tjScaled( 12, weight: .medium))
|
|
.foregroundStyle(Tj.Palette.text3)
|
|
}
|
|
HStack(spacing: 8) {
|
|
Text(Self.relativeDate(export.createdAt))
|
|
.font(.tjScaled( 11))
|
|
.foregroundStyle(Tj.Palette.text3)
|
|
if export.decodeRate > 0 {
|
|
Text(String(format: "%.1f tok/s", export.decodeRate))
|
|
.font(.tjScaled( 10, design: .monospaced))
|
|
.foregroundStyle(Tj.Palette.leaf)
|
|
}
|
|
Spacer()
|
|
if let label = export.inferredLabelCN ?? export.inferredIntent {
|
|
TjBadge(text: label, style: .neutral)
|
|
}
|
|
}
|
|
}
|
|
.padding(14)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.tjCard()
|
|
}
|
|
|
|
/// 复用单个 formatter:RelativeDateTimeFormatter 初始化较贵,列表每行每次重绘都 new 会累积开销。
|
|
/// 用系统 Locale.current(与原实现一致),进程内不变,可安全缓存。
|
|
private static let relativeFormatter: RelativeDateTimeFormatter = {
|
|
let f = RelativeDateTimeFormatter()
|
|
f.locale = Locale.current
|
|
f.unitsStyle = .full
|
|
return f
|
|
}()
|
|
|
|
static func relativeDate(_ d: Date) -> String {
|
|
relativeFormatter.localizedString(for: d, relativeTo: .now)
|
|
}
|
|
}
|
|
|
|
#Preview {
|
|
NavigationStack {
|
|
HealthExportListView()
|
|
}
|
|
.modelContainer(for: [
|
|
Indicator.self, Report.self, DiaryEntry.self, Asset.self,
|
|
ChatTurn.self, Symptom.self, UserProfile.self,
|
|
MetricReminder.self, CustomMonitorMetric.self, HealthExport.self
|
|
], inMemory: true)
|
|
}
|