Voller Marketplace-Flow mit TabBar und Universal-Link-Handler. Drei Live-Decks (Geografie, English A2, Periodensystem) sind browse-, abonnier- und lernbar. - PublicDeckEntry/PublicDeck/PublicDeckVersion/PublicDeckOwner/ PublicDeckDetail Codable mit snake_case - ExploreResponse, BrowseResponse, SubscribeResponse - MarketplaceSort-Enum (recent/popular/trending) - CardsAPI.explore/browseMarketplace/publicDeck/subscribe/unsubscribe - MarketplaceStore @Observable mit Explore + Browse States - ExploreView: Featured + Trending Horizontal-Carousels, Browse-Link - BrowseView: Searchable + Sort-Picker + List - PublicDeckView: Header/Metadata/Subscribe — Subscribe löst Auto-Fork serverseitig aus, Response liefert private_deck_id, NavigationLink zum eigenen Deck - PublicDeckCard + BrowseRow mit forest-Theme - RootView: TabBar (Decks/Entdecken/Account) statt Single-View - Universal-Link-Handler: onOpenURL + onContinueUserActivity für https://cardecky.mana.how/d/<slug> und cards://d/<slug> - associated-domains: applinks:cardecky.mana.how im entitlement - 5 neue Marketplace-Decoding-Tests (35 Total grün) Universal-Links funktionieren erst nach AASA-Setup auf cardecky.mana.how/.well-known/apple-app-site-association (Web-Aufgabe, heute 404). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
162 lines
4.6 KiB
Swift
162 lines
4.6 KiB
Swift
import Foundation
|
|
|
|
/// Browse-Eintrag aus `/api/v1/marketplace/decks` und `.../explore`.
|
|
struct PublicDeckEntry: Codable, Hashable, Sendable, Identifiable {
|
|
let slug: String
|
|
let title: String
|
|
let description: String?
|
|
let language: String?
|
|
let category: String?
|
|
let license: String
|
|
let priceCredits: Int
|
|
let cardCount: Int
|
|
let starCount: Int
|
|
let subscriberCount: Int
|
|
let isFeatured: Bool
|
|
let createdAt: Date
|
|
let owner: PublicDeckOwner
|
|
|
|
var id: String { slug }
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case slug, title, description, language, category, license
|
|
case priceCredits = "price_credits"
|
|
case cardCount = "card_count"
|
|
case starCount = "star_count"
|
|
case subscriberCount = "subscriber_count"
|
|
case isFeatured = "is_featured"
|
|
case createdAt = "created_at"
|
|
case owner
|
|
}
|
|
|
|
var isPaid: Bool { priceCredits > 0 }
|
|
}
|
|
|
|
struct PublicDeckOwner: Codable, Hashable, Sendable {
|
|
let slug: String
|
|
let displayName: String
|
|
let verifiedMana: Bool
|
|
let verifiedCommunity: Bool
|
|
let pseudonym: String?
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case slug
|
|
case displayName = "display_name"
|
|
case verifiedMana = "verified_mana"
|
|
case verifiedCommunity = "verified_community"
|
|
case pseudonym
|
|
}
|
|
|
|
init(from decoder: Decoder) throws {
|
|
let c = try decoder.container(keyedBy: CodingKeys.self)
|
|
slug = try c.decode(String.self, forKey: .slug)
|
|
displayName = try c.decode(String.self, forKey: .displayName)
|
|
verifiedMana = try c.decode(Bool.self, forKey: .verifiedMana)
|
|
verifiedCommunity = try c.decode(Bool.self, forKey: .verifiedCommunity)
|
|
pseudonym = try c.decodeIfPresent(String.self, forKey: .pseudonym)
|
|
}
|
|
}
|
|
|
|
/// Response von `GET /api/v1/marketplace/explore`.
|
|
struct ExploreResponse: Decodable, Sendable {
|
|
let featured: [PublicDeckEntry]
|
|
let trending: [PublicDeckEntry]
|
|
}
|
|
|
|
/// Response von `GET /api/v1/marketplace/decks`.
|
|
struct BrowseResponse: Decodable, Sendable {
|
|
let items: [PublicDeckEntry]
|
|
let total: Int
|
|
}
|
|
|
|
/// Vollständiges Public-Deck aus `GET /api/v1/marketplace/decks/:slug`.
|
|
struct PublicDeck: Codable, Hashable, Sendable, Identifiable {
|
|
let id: String
|
|
let slug: String
|
|
let title: String
|
|
let description: String?
|
|
let language: String?
|
|
let category: String?
|
|
let license: String
|
|
let priceCredits: Int
|
|
let ownerUserId: String
|
|
let latestVersionId: String?
|
|
let isFeatured: Bool
|
|
let isTakedown: Bool
|
|
let createdAt: Date
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case id, slug, title, description, language, category, license
|
|
case priceCredits = "price_credits"
|
|
case ownerUserId = "owner_user_id"
|
|
case latestVersionId = "latest_version_id"
|
|
case isFeatured = "is_featured"
|
|
case isTakedown = "is_takedown"
|
|
case createdAt = "created_at"
|
|
}
|
|
}
|
|
|
|
struct PublicDeckVersion: Codable, Hashable, Sendable, Identifiable {
|
|
let id: String
|
|
let deckId: String
|
|
let semver: String
|
|
let changelog: String?
|
|
let contentHash: String?
|
|
let cardCount: Int
|
|
let publishedAt: Date
|
|
let deprecatedAt: Date?
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case id
|
|
case deckId = "deck_id"
|
|
case semver
|
|
case changelog
|
|
case contentHash = "content_hash"
|
|
case cardCount = "card_count"
|
|
case publishedAt = "published_at"
|
|
case deprecatedAt = "deprecated_at"
|
|
}
|
|
}
|
|
|
|
/// Response von `GET /api/v1/marketplace/decks/:slug`.
|
|
struct PublicDeckDetail: Decodable, Sendable {
|
|
let deck: PublicDeck
|
|
let latestVersion: PublicDeckVersion?
|
|
let owner: PublicDeckOwner?
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case deck
|
|
case latestVersion = "latest_version"
|
|
case owner
|
|
}
|
|
}
|
|
|
|
/// Response von `POST /api/v1/marketplace/decks/:slug/subscribe`.
|
|
struct SubscribeResponse: Decodable, Sendable {
|
|
let subscribed: Bool
|
|
let deckSlug: String
|
|
let currentVersionId: String?
|
|
let privateDeckId: String
|
|
|
|
enum CodingKeys: String, CodingKey {
|
|
case subscribed
|
|
case deckSlug = "deck_slug"
|
|
case currentVersionId = "current_version_id"
|
|
case privateDeckId = "private_deck_id"
|
|
}
|
|
}
|
|
|
|
/// Browse-Sort-Optionen aus `BrowseQuerySchema`.
|
|
enum MarketplaceSort: String, Sendable, CaseIterable {
|
|
case recent
|
|
case popular
|
|
case trending
|
|
|
|
var label: String {
|
|
switch self {
|
|
case .recent: "Neueste"
|
|
case .popular: "Beliebt"
|
|
case .trending: "Im Trend"
|
|
}
|
|
}
|
|
}
|