85bfdf2b56
New 'Recap' phase — turns speakers.json into a human-readable recap, leveraging recap-relay's proven logic/prompts but calling the Spark gateway's OpenAI-compatible /v1/chat/completions directly (same host/TLS as label-merge; Qwen3-35B). We start from already-named speakers (label-merge), so recap-relay's speaker clustering + name-inference are skipped entirely. - GatewayLLMClient: /v1/chat/completions (JSON mode), model discovery via /api/endpoints, TLS-skip reuse, 503 retry, sequential. - RecapAnalyzer: speakers.json → numbered [N] (MM:SS) Name: text transcript → time-windowed analyze (single window for short calls, 18min/2min overlap for long) → stitch/dedup topic sections → meeting extras (TLDR/decisions/action_items/ open_questions/key_quotes). Defensive JSON parsing of LLM output. - RecapRenderer: writes transcript.md + a self-contained dark-theme recap.html (topic sections w/ collapsible transcripts, extras panels, speaker color chips, full timestamped speaker-attributed transcript, print styles). - SessionController.buildRecap: best-effort after speakers.json (gated by settings.recapEnabled); surfaces recapURL → menu 'Open recap'. Skips silently if the gateway has no LLM. Settings toggle added. Validated END-TO-END on the real Meet session against the live gateway: dual-channel transcription → 3 topic sections + accurate TLDR + key quotes; 'Go Bitcoin' correctly attributed to the remote speaker. 46/46 XCTest (10 new).
85 lines
3.2 KiB
Swift
85 lines
3.2 KiB
Swift
import SwiftUI
|
|
import AppKit
|
|
|
|
/// Settings panel (pushed from the menu-bar panel).
|
|
struct SettingsView: View {
|
|
@EnvironmentObject private var settings: AppSettings
|
|
|
|
var body: some View {
|
|
Form {
|
|
Section("SparkControl backend") {
|
|
TextField("Base URL", text: $settings.backendBaseURL)
|
|
.textFieldStyle(.roundedBorder)
|
|
Toggle("Skip TLS verification (self-signed cert)",
|
|
isOn: $settings.skipTLSVerification)
|
|
}
|
|
|
|
Section("Call detection") {
|
|
Toggle("Auto-record when a call is detected", isOn: $settings.autoRecordOnDetection)
|
|
Text("Detects Zoom, Teams, Signal, and Google Meet (any browser).")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
Section("Transcription") {
|
|
TextField("Your name", text: $settings.selfName)
|
|
.textFieldStyle(.roundedBorder)
|
|
Toggle("Auto-send recordings to backend", isOn: $settings.autoSendOnStop)
|
|
Toggle("Build readable recap (topics + highlights)", isOn: $settings.recapEnabled)
|
|
Text("Your name labels your mic channel. Auto-send transcribes on stop; the recap writes transcript.md + recap.html.")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
Section("Output") {
|
|
HStack {
|
|
Text(settings.outputFolderPath)
|
|
.lineLimit(1)
|
|
.truncationMode(.middle)
|
|
.foregroundStyle(.secondary)
|
|
Spacer()
|
|
Button("Choose…", action: chooseFolder)
|
|
}
|
|
}
|
|
|
|
Section("Adapters") {
|
|
Text("Inert in Phase 0 — these toggles only persist for now.")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
ForEach(AppSettings.adapterKeys, id: \.key) { adapter in
|
|
Toggle(adapter.label, isOn: binding(for: adapter.key))
|
|
}
|
|
}
|
|
}
|
|
.formStyle(.grouped)
|
|
.frame(width: 320)
|
|
.navigationTitle("Settings")
|
|
}
|
|
|
|
private func binding(for key: String) -> Binding<Bool> {
|
|
Binding(
|
|
get: { settings.adapterEnabled[key] ?? true },
|
|
set: { settings.adapterEnabled[key] = $0 }
|
|
)
|
|
}
|
|
|
|
private func chooseFolder() {
|
|
let panel = NSOpenPanel()
|
|
panel.canChooseDirectories = true
|
|
panel.canChooseFiles = false
|
|
panel.allowsMultipleSelection = false
|
|
panel.prompt = "Choose"
|
|
panel.directoryURL = settings.outputFolderURL
|
|
|
|
// The app is a menu-bar accessory and this is invoked from the transient
|
|
// MenuBarExtra(.window) popover. Use the async begin(...) API rather than
|
|
// runModal() — a nested modal loop can let the popover dismiss the panel
|
|
// out from under it. Activate first so the panel comes to the front.
|
|
NSApp.activate(ignoringOtherApps: true)
|
|
panel.begin { response in
|
|
guard response == .OK, let url = panel.url else { return }
|
|
settings.outputFolderPath = url.path
|
|
}
|
|
}
|
|
}
|