zitare-native/Sources/App/ZitareNativeApp.swift
Till JS 53f8043a2d ζ-3.5b: Pending-Queue-UI + NWPathMonitor-Reconnect-Flush
Pending-Queue-UI:
- AccountView.pendingQueueCard listet alle wartenden Submissions
  mit Text-Preview (120c), Author, createdAt, retryCount, lastError
- "Jetzt versuchen"-Button triggert tryFlush(api:)
- Trash-Icon pro Row löscht einzeln aus der Queue
- Pull-to-Refresh aktualisiert beide Listen
- Card-View nur sichtbar wenn Queue-Tiefe > 0

ReachabilityWatcher:
- NWPathMonitor erkennt Reconnect-Flanke (offline → online)
- Bei Reconnect: SubmissionQueue.tryFlush auf @MainActor
- Filtert reine Wifi↔Mobil-Switches raus (nur "wieder erreichbar"
  zählt)
- Lebt im App-Root, startet nach Launch via .task

Files:
- Sources/Core/Submit/ReachabilityWatcher.swift (neu)
- Sources/App/ZitareNativeApp.swift: reachability.start in .task
- Sources/Features/Account/AccountView.swift: pendingQueueCard

iOS + macOS BUILD SUCCEEDED.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-19 16:26:55 +02:00

94 lines
3.3 KiB
Swift

import ManaAuthUI
import ManaCore
import SwiftData
import SwiftUI
import WidgetKit
@main
struct ZitareNativeApp: App {
@State private var auth: AuthClient
@State private var authGate: ManaAuthGate
@State private var submissionQueue: SubmissionQueue
private let snapshotContainer: ModelContainer?
private let reachability = ReachabilityWatcher()
init() {
let auth = AuthClient(config: AppConfig.manaAppConfig)
auth.bootstrap()
_auth = State(initialValue: auth)
_authGate = State(initialValue: ManaAuthGate(auth: auth))
do {
snapshotContainer = try SnapshotContainer.make()
} catch {
Log.snapshot.error(
"SnapshotContainer init fehlgeschlagen: \(String(describing: error), privacy: .public)"
)
snapshotContainer = nil
}
let pending: ModelContainer
do {
pending = try PendingSubmissionContainer.make()
} catch {
Log.app.error(
"PendingSubmissionContainer-Disk init fehlgeschlagen, falle auf in-memory zurück: \(String(describing: error), privacy: .public)"
)
// In-memory-Fallback statt nil. Submissions sind dann nur
// für die aktuelle Session persistiert besser als Crash.
pending = try! PendingSubmissionContainer.make(inMemory: true)
}
_submissionQueue = State(initialValue: SubmissionQueue(container: pending))
Log.app.info(
"Zitare starting — auth status: \(String(describing: auth.status), privacy: .public)"
)
}
var body: some Scene {
WindowGroup {
RootView()
.environment(auth)
.environment(authGate)
.environment(submissionQueue)
.tint(ZitareTheme.primary)
.task {
await refreshSnapshot()
await flushPending()
startReachability()
}
}
}
private func flushPending() async {
let api = ZitareAPI(auth: auth)
let sent = await submissionQueue.tryFlush(api: api)
if sent > 0 {
Log.app.info("Auto-flushed \(sent) pending submission(s) at launch")
}
}
/// Startet den NWPathMonitor und hooked Reconnect Flush.
/// `Task.detached` + `await MainActor`, weil das Reconnect-Closure
/// vom Network-Framework auf einer privaten Queue feuert.
private func startReachability() {
let queue = submissionQueue
let auth = auth
reachability.start { [weak queue] in
guard let queue else { return }
Task { @MainActor in
let api = ZitareAPI(auth: auth)
let sent = await queue.tryFlush(api: api)
if sent > 0 {
Log.app.info("Reachability-Flush: \(sent) Submission(s) nachgereicht")
}
}
}
}
private func refreshSnapshot() async {
guard let container = snapshotContainer else { return }
let sync = SnapshotSync(container: container)
await sync.tryRefresh()
// Widget-Timeline neu erstellen lassen, sodass der nächste
// Render-Pass den frischen Snapshot sieht.
WidgetCenter.shared.reloadAllTimelines()
}
}