Vervollständigt die Cardecky-Web-Parität für Deck- und Card-Workflows. γ-1+γ-2 (AI-Deck-Generierung) - 4-Modi-Picker im DeckEditorView Create-Sheet: Leer/KI/Bild/CSV - POST /api/v1/decks/generate für Text-Prompt + 10/min Rate-Limit-UI - POST /api/v1/decks/from-image mit PhotosPicker + PDF-Importer (max 5 Files, 10 MiB/Bild, 30 MiB/PDF), Multipart-Body in CardsAPI+Generation - Loading-Overlay mit Task-Cancellation, Error-Mapping für 429/413/502 γ-3 (Card-Edit) - CardEditorView mit Mode .create(deckId:) / .edit(card:) - Image-Occlusion + Audio-Front behalten bestehenden Media-Ref, solange User nicht ersetzt — MediaCache lädt Bild nach - Type-Picker im Edit-Modus aus (Server-immutable) - CardEditorPayload + CardEditorMediaFields als Sub-Views γ-4 (Pull-Update + Duplicate + Archive) - POST /marketplace/private/:id/pull-update mit Smart-Merge-Anzeige - POST /decks/:id/duplicate - Archive-Toggle im Edit-Modus, Server filtert Liste serverseitig - DeckSecondaryActions als eigenes Sub-View γ-6 (CSV-Import) - RFC-4180-ish Parser (Quote-Escape, Header-Detect, BOM-strip) - Preview-Liste + sequentielle Card-Inserts mit Live-Progress - Image-Occlusion/Audio-Front werden geskipped (UI flaggt) γ-7 (Marketplace-Publish) + Follow-up (Report + Block + Re-Publish) - MarketplacePublishView mit lazy Author-Setup + Init + Publish 1.0.0 - Re-Publish-Modus: Picker für eigene Marketplace-Decks + Auto-Semver-Bump (Minor +1) - MarketplaceCardConverter (typing → type-in, audio-front → skipped, image-occlusion → skipped — Server hat keinen MP-Media-Re-Upload) - Toolbar-Menü auf PublicDeckView: „Deck melden …" + Author-Blockieren (App-Store-Guideline 5.1.1(v)) - ReportDeckSheet mit Reason-Picker (6 Kategorien) + optional Message - BlockedAuthorsView in Settings mit Swipe-Entblocken γ-8 (PDF-Export) - DeckPrintView mit SFSafariViewController auf cardecky.mana.how/decks/:id/print — iOS Share-Sheet → PDF speichern Side-Fixes (mid-stream) - StudySessionView: Card-Aspect-Ratio springt nicht mehr beim Flip (Bottom-Bar in ZStack fixer Höhe) - RootView: Glass-Pille für „Neues Deck"-Accessory + .guest- und .twoFactorRequired-Cases nachgezogen - DeckListView: Account-Toolbar-Button entfernt (Account-Tab unten ist alleinige Anlaufstelle) Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
114 lines
4.1 KiB
Swift
114 lines
4.1 KiB
Swift
import Foundation
|
||
import ManaCore
|
||
|
||
/// AI-Deck-Generierung + Multipart-Helpers — ausgelagert aus `CardsAPI`,
|
||
/// damit der Haupt-Actor unter der Type-Body-Length-Grenze bleibt.
|
||
extension CardsAPI {
|
||
/// `POST /api/v1/decks/generate` — KI generiert Deck aus Prompt.
|
||
/// Rate-Limit serverseitig 10/min. Antwort dauert typisch 10–60s
|
||
/// (synchron, kein Streaming).
|
||
func generateDeckFromText(_ body: DeckGenerateBody) async throws -> DeckGenerateResponse {
|
||
let data = try makeJSON(body)
|
||
let (responseData, http) = try await transport.request(
|
||
path: "/api/v1/decks/generate",
|
||
method: "POST",
|
||
body: data
|
||
)
|
||
try ensureOK(http, data: responseData)
|
||
return try decoder.decode(DeckGenerateResponse.self, from: responseData)
|
||
}
|
||
|
||
/// `POST /api/v1/decks/from-image` — Vision-LLM generiert Deck aus
|
||
/// Bildern und/oder PDFs (max 5 Files, 10 MiB pro Bild, 30 MiB pro PDF)
|
||
/// und optional einer URL für Zusatz-Kontext. Rate-Limit 10/min.
|
||
/// Multipart-Body mit `file`-Parts (wiederholt) + Text-Felder.
|
||
func generateDeckFromMedia(
|
||
files: [GenerationMediaFile],
|
||
language: GenerationLanguage,
|
||
count: Int,
|
||
url: String?
|
||
) async throws -> DeckGenerateResponse {
|
||
let boundary = "cards-native-\(UUID().uuidString)"
|
||
let body = makeFromImageMultipartBody(
|
||
files: files,
|
||
language: language,
|
||
count: count,
|
||
url: url,
|
||
boundary: boundary
|
||
)
|
||
let (responseData, http) = try await transport.request(
|
||
path: "/api/v1/decks/from-image",
|
||
method: "POST",
|
||
body: body,
|
||
contentType: "multipart/form-data; boundary=\(boundary)"
|
||
)
|
||
try ensureOK(http, data: responseData)
|
||
return try decoder.decode(DeckGenerateResponse.self, from: responseData)
|
||
}
|
||
|
||
// MARK: - Multipart
|
||
|
||
/// Single-File-Multipart-Body für `/media/upload`.
|
||
func makeMultipartBody(
|
||
file: Data,
|
||
filename: String,
|
||
mimeType: String,
|
||
boundary: String
|
||
) -> Data {
|
||
var body = Data()
|
||
let lineBreak = "\r\n"
|
||
let header = """
|
||
--\(boundary)\(lineBreak)\
|
||
Content-Disposition: form-data; name="file"; filename="\(filename)"\(lineBreak)\
|
||
Content-Type: \(mimeType)\(lineBreak)\(lineBreak)
|
||
"""
|
||
body.append(Data(header.utf8))
|
||
body.append(file)
|
||
body.append(Data(lineBreak.utf8))
|
||
body.append(Data("--\(boundary)--\(lineBreak)".utf8))
|
||
return body
|
||
}
|
||
|
||
/// Multi-File-Multipart-Body für `/decks/from-image` — mehrere Files
|
||
/// unter dem Form-Feld `file` (Server liest sie via `getAll('file')`)
|
||
/// plus optional `language`, `count`, `url` als Text-Felder.
|
||
func makeFromImageMultipartBody(
|
||
files: [GenerationMediaFile],
|
||
language: GenerationLanguage,
|
||
count: Int,
|
||
url: String?,
|
||
boundary: String
|
||
) -> Data {
|
||
var body = Data()
|
||
let lineBreak = "\r\n"
|
||
|
||
func appendField(name: String, value: String) {
|
||
let part = """
|
||
--\(boundary)\(lineBreak)\
|
||
Content-Disposition: form-data; name="\(name)"\(lineBreak)\(lineBreak)\
|
||
\(value)\(lineBreak)
|
||
"""
|
||
body.append(Data(part.utf8))
|
||
}
|
||
|
||
appendField(name: "language", value: language.rawValue)
|
||
appendField(name: "count", value: String(count))
|
||
if let url, !url.trimmingCharacters(in: .whitespaces).isEmpty {
|
||
appendField(name: "url", value: url)
|
||
}
|
||
|
||
for file in files {
|
||
let header = """
|
||
--\(boundary)\(lineBreak)\
|
||
Content-Disposition: form-data; name="file"; filename="\(file.filename)"\(lineBreak)\
|
||
Content-Type: \(file.mimeType)\(lineBreak)\(lineBreak)
|
||
"""
|
||
body.append(Data(header.utf8))
|
||
body.append(file.data)
|
||
body.append(Data(lineBreak.utf8))
|
||
}
|
||
|
||
body.append(Data("--\(boundary)--\(lineBreak)".utf8))
|
||
return body
|
||
}
|
||
}
|