wordeck-native/Sources/Features/Decks/PendingShareConsumeView.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

116 lines
3.9 KiB
Swift

import ManaCore
import SwiftData
import SwiftUI
struct PendingShareRoute: Hashable {
let share: PendingShare
}
/// User wählt Ziel-Deck, optional Back-Text. Submit CardCreate
/// via API, anschließend wird die PendingShare aus dem Store entfernt.
struct PendingShareConsumeView: View {
let share: PendingShare
let onDone: () -> Void
@Environment(AuthClient.self) private var auth
@Environment(\.dismiss) private var dismiss
@Query(sort: \CachedDeck.updatedAt, order: .reverse) private var decks: [CachedDeck]
@State private var selectedDeckId: String?
@State private var front: String
@State private var back: String = ""
@State private var isSubmitting = false
@State private var errorMessage: String?
init(share: PendingShare, onDone: @escaping () -> Void) {
self.share = share
self.onDone = onDone
_front = State(initialValue: share.text)
}
var body: some View {
Form {
Section("Ziel-Deck") {
if decks.isEmpty {
Text("Erst ein Deck erstellen.")
.foregroundStyle(WordeckTheme.mutedForeground)
} else {
Picker("Deck", selection: $selectedDeckId) {
Text("Wählen …").tag(String?.none)
ForEach(decks) { deck in
Text(deck.name).tag(String?.some(deck.id))
}
}
}
}
Section("Vorderseite") {
TextField("Front", text: $front, axis: .vertical)
.lineLimit(2 ... 6)
}
Section("Rückseite") {
TextField("Back (optional, default: Quell-URL)", text: $back, axis: .vertical)
.lineLimit(2 ... 4)
}
if let sourceURL = share.sourceURL {
Section("Quelle") {
Text(sourceURL)
.font(.caption)
.foregroundStyle(WordeckTheme.mutedForeground)
}
}
if let errorMessage {
Section {
Text(errorMessage)
.font(.footnote)
.foregroundStyle(WordeckTheme.error)
}
}
}
.navigationTitle("Geteilten Inhalt speichern")
#if os(iOS)
.navigationBarTitleDisplayMode(.inline)
#endif
.toolbar {
ToolbarItem(placement: .cancellationAction) {
Button("Verwerfen", role: .destructive) {
PendingShareStore.remove(id: share.id)
onDone()
dismiss()
}
}
ToolbarItem(placement: .confirmationAction) {
Button("Speichern") { Task { await submit() } }
.disabled(selectedDeckId == nil || front.trimmed.isEmpty || isSubmitting)
}
}
}
private func submit() async {
guard let deckId = selectedDeckId else { return }
isSubmitting = true
errorMessage = nil
defer { isSubmitting = false }
let backText = back.trimmed.isEmpty ? (share.sourceURL ?? "") : back.trimmed
let api = WordeckAPI(auth: auth)
let body = CardCreateBody(
deckId: deckId,
type: .basic,
fields: CardFieldsBuilder.basic(front: front.trimmed, back: backText),
mediaRefs: nil
)
do {
_ = try await api.createCard(body)
onDone()
dismiss()
} catch {
errorMessage = (error as? LocalizedError)?.errorDescription ?? String(describing: error)
}
}
}
private extension String {
var trimmed: String {
trimmingCharacters(in: .whitespacesAndNewlines)
}
}