wordeck-native/Sources/Features/Study/StudySession.swift
Till JS 542082772a refactor(big-bang): cards-native → wordeck-native
Code + Identity-Rename zur Vorbereitung auf Apple-Dev-Portal-Aktion
(Bundle ev.mana.wordeck, App-Group group.ev.mana.wordeck, AASA
applinks:wordeck.com). Build bleibt funktional, aber gegen die
neue text-only-API können image-occlusion-Creates 422 zurückgeben —
das wird mit der Wordeck-Native v1.0-Welle (parallele Apple-Aktion)
sauber gemacht.

Umbenennung:
- 41 Files: cardecky/Cardecky → wordeck/Wordeck (Display, Strings,
  Kommentare)
- 57 Files: CardsNative → WordeckNative, CardsAPI → WordeckAPI,
  CardsTheme → WordeckTheme, CardsBrand → WordeckBrand, CardsWidget →
  WordeckWidget, CardsDueWidget → WordeckDueWidget
- Bundle-ID ev.mana.cardecky → ev.mana.wordeck (project.yml,
  Info.plist, entitlements, Keychain-Service, App-Group)
- AASA applinks:cardecky.mana.how → applinks:wordeck.com
- API-Base cardecky-api.mana.how → api.wordeck.com
- 10 Files renamed (App-Entry, API-Extensions, Theme, Widget,
  Entitlements, Tests)
- xcodeproj regenerated via xcodegen → WordeckNative.xcodeproj
- MaskRegionsTests.swift gelöscht (image-occlusion entfällt mit
  Wordeck text-only)

Forgejo-Repo git.mana.how/till/cards-native → wordeck-native umbenannt
(Auto-Redirect aktiv). Lokales Verzeichnis Code/cards-native/ bleibt
vorerst — wird beim nächsten Apple-Setup mit Bundle-Test umbenannt.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-17 23:10:42 +02:00

95 lines
2.7 KiB
Swift

import Foundation
import ManaCore
import Observation
import SwiftData
/// State-Machine für eine Lern-Session. Lädt Due-Reviews beim Start,
/// rendert eine Karte nach der anderen, schickt Grades via GradeQueue ab.
@MainActor
@Observable
final class StudySession {
enum Phase {
case loading
case studying
case finished
case failed(String)
}
private(set) var phase: Phase = .loading
private(set) var queue: [DueReview] = []
private(set) var currentIndex: Int = 0
private(set) var isFlipped: Bool = false
private(set) var totalGraded: Int = 0
let deckId: String
let deckName: String
private let api: WordeckAPI
private let gradeQueue: GradeQueue
init(deckId: String, deckName: String, auth: AuthClient, context: ModelContext) {
self.deckId = deckId
self.deckName = deckName
api = WordeckAPI(auth: auth)
gradeQueue = GradeQueue(api: api, context: context)
}
var current: DueReview? {
guard queue.indices.contains(currentIndex) else { return nil }
return queue[currentIndex]
}
var remaining: Int {
max(0, queue.count - currentIndex)
}
func start() async {
phase = .loading
do {
queue = try await api.dueReviews(deckId: deckId, limit: 500)
currentIndex = 0
isFlipped = false
totalGraded = 0
if queue.isEmpty {
phase = .finished
} else {
phase = .studying
}
let count = queue.count
let id = deckId
Log.study.info("Session start — \(count, privacy: .public) due in deck \(id, privacy: .public)")
} catch {
let msg = (error as? LocalizedError)?.errorDescription ?? String(describing: error)
phase = .failed(msg)
Log.study.error("Session start failed: \(msg, privacy: .public)")
}
}
func flip() {
guard case .studying = phase else { return }
isFlipped.toggle()
}
func grade(_ rating: Rating) async {
guard case .studying = phase, let card = current else { return }
let reviewedAt = Date.now
await gradeQueue.submit(
cardId: card.review.cardId,
subIndex: card.review.subIndex,
rating: rating,
reviewedAt: reviewedAt
)
totalGraded += 1
advance()
}
private func advance() {
currentIndex += 1
isFlipped = false
if currentIndex >= queue.count {
phase = .finished
let count = totalGraded
Log.study.info("Session finished — graded \(count, privacy: .public)")
}
}
}