Files
hyungi_document_server/Sources/AppFeature/Pages/MemosView.swift
T
Hyungi 560efb9554 feat(s3): SwiftUI sage 3-pane shell + 6 pages + AI seam
- AppFeature: SageTheme tokens, AppModel (@MainActor @Observable store), RootView (DEVONthink NavigationSplitView), Dashboard/Documents(MD-first+pending fallback+?token= download)/Search/Ask/Memos/Digest pages
- AI seam: AIService actor + AIResult, AppAIComposition (MockAIProvider x4 tiers), AICompletionView (numbered citations + always-visible routing badge), backend picker with visible explicit-unavailable error
- MarkdownView: block-aware renderer (GFM table separator-row skip, AttributedString inline-only)
- DSApp: thin @main, injects FixtureDSClient + mock AIRouter (zero backend / zero LLM)

swift build (full app) + swift test (19) green under Swift 6 strict concurrency. Sources/AI untouched (isolation vs freeze 17f8830 = clean).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-04 17:26:02 +09:00

77 lines
2.9 KiB
Swift

import SwiftUI
import DSKit
/// Capture-first treatment: quick-capture box + list distinguishing note vs audio.
struct MemoListView: View {
@Environment(AppModel.self) private var model
@State private var draft: String = ""
var body: some View {
VStack(alignment: .leading, spacing: 0) {
HStack(spacing: 8) {
TextField("빠른 메모 캡처", text: $draft)
.textFieldStyle(.roundedBorder)
Button("저장") {
let content = draft
draft = ""
Task { _ = try? await model.client.createMemo(MemoCreate(content: content)) }
}
.buttonStyle(.bordered)
.disabled(draft.isEmpty)
}
.padding(12)
let selection = Binding<Int?>(
get: { model.selectedMemoID },
set: { if let id = $0 { Task { await model.openMemo(id) } } }
)
List(model.memoList, selection: selection) { memo in
VStack(alignment: .leading, spacing: 3) {
HStack(spacing: 6) {
if memo.isAudio { Chip("음성", Sage.formatColor("audio")) }
if memo.aiEventKind == "task" { Chip("할 일", Sage.amber) }
Text(memo.title ?? "메모 \(memo.id)")
.font(.callout.weight(.medium)).foregroundStyle(Sage.ink).lineLimit(1)
Spacer()
if memo.isPinned { Text("고정").font(.caption2).foregroundStyle(Sage.amber) }
}
Text(memo.content ?? "").font(.caption).foregroundStyle(Sage.muted).lineLimit(2)
}
.padding(.vertical, 3)
}
.listStyle(.inset)
}
.background(Sage.surface)
}
}
struct MemoDetailView: View {
let memo: MemoResponse
var body: some View {
ScrollView {
VStack(alignment: .leading, spacing: 12) {
Text(memo.title ?? "메모").font(.title2.weight(.bold)).foregroundStyle(Sage.ink)
if memo.isAudio {
HStack(spacing: 8) {
Image(systemName: "waveform")
Text("원본 재생 (스캐폴드에서는 비활성)").foregroundStyle(Sage.muted)
}
.padding(10)
.frame(maxWidth: .infinity, alignment: .leading)
.background(Sage.surface, in: RoundedRectangle(cornerRadius: 8))
}
if let tags = memo.aiTags, !tags.isEmpty {
HStack(spacing: 6) { ForEach(tags, id: \.self) { Chip($0, Sage.brand) } }
}
MarkdownView(memo.content ?? "")
}
.padding(20)
}
.background(Sage.surface)
}
}