chahinebrini 685782b538 fix(coach): dynamische Sprache (Text-Detection + App-Locale-Fallback)
LLM-Prompt (message.post + sos-stream):
- LANG_INSTRUCTIONS Map raus, ersetzt durch dynamische Instruktion
  'Reply in {detectedFromUser} ... fallback: {appLang}'
- Lyra matcht jetzt die Sprache der letzten User-Message (per
  detectLang Unicode-Detection); App-Locale ist nur noch Fallback
- Instruktion doppelt eingehängt (Anfang + Ende des System-Prompts)
  gegen recency bias bei langen deutschen Prompts

TTS (speak dispatcher + speak-cartesia + speak-elevenlabs):
- Kein 'de'-Default mehr für language. detectLang(text, locale) leitet
  Sprache primär aus dem Antwort-Text ab (Arabic/Cyrillic/CJK/Turkish-
  Letters), Locale als Fallback
- Cartesia + ElevenLabs: language/language_code nur senden wenn
  ableitbar, sonst Provider auto-detect statt erzwungenem 'de'
- speak-cartesia: sonic-2 → sonic-3 (Multi-Lang, war beim Dispatcher-
  Fix gestern vergessen worden)
- Google: en-US neutraler Fallback statt de-DE-Bias

Neu: server/utils/detect-lang.ts
2026-05-31 00:12:40 +02:00

99 lines
3.5 KiB
Swift

import SwiftUI
struct PreflightView: View {
@Environment(WizardModel.self) private var model
@State private var fmiConfirmed = false
@State private var sdpConfirmed = false
@State private var appleIdConfirmed = false
@State private var rebreakAppInstalled = false
var body: some View {
ScrollView {
VStack(alignment: .leading, spacing: 24) {
header
Text("Bevor wir dein iPhone supervisieren, müssen ein paar Apple-Sicherheitschecks erledigt sein. Hak die Punkte ab, sobald du sie auf dem iPhone gemacht hast.")
.foregroundStyle(.secondary)
checklist
Spacer()
navigationBar
}
.padding(40)
}
}
private var header: some View {
HStack {
Image(systemName: "checklist")
.font(.system(size: 30))
.foregroundStyle(.tint)
Text("Pre-Flight Check")
.font(.title).bold()
}
}
private var checklist: some View {
VStack(spacing: 12) {
checklistItem(
checked: $fmiConfirmed,
title: "Find My iPhone deaktiviert",
detail: "Settings → [Apple-ID] → Wo ist? → Mein iPhone suchen → AUS. Ohne das blockiert Apple das Supervisieren (ErrorCode 211)."
)
checklistItem(
checked: $sdpConfirmed,
title: "Stolen Device Protection ausgeschaltet",
detail: "Settings → Face ID & Code → Schutz für gestohlene Geräte → AUS. SDP zwingt FMI an — muss VOR FMI-Toggle aus."
)
checklistItem(
checked: $appleIdConfirmed,
title: "Apple-ID-Passwort griffbereit",
detail: "Apple fragt evtl. dein Apple-ID-PW während des FMI-Toggles ab. Halte es bereit."
)
checklistItem(
checked: $rebreakAppInstalled,
title: "ReBreak-App ist auf dem iPhone installiert",
detail: "Über TestFlight (https://testflight.apple.com/join/...). Erst danach kann der Wizard die App in den Managed-State versetzen."
)
}
}
private func checklistItem(checked: Binding<Bool>, title: String, detail: String) -> some View {
Button(action: { checked.wrappedValue.toggle() }) {
HStack(alignment: .top, spacing: 12) {
Image(systemName: checked.wrappedValue ? "checkmark.square.fill" : "square")
.font(.title3)
.foregroundStyle(checked.wrappedValue ? Color.accentColor : Color.secondary)
VStack(alignment: .leading, spacing: 4) {
Text(title).font(.headline)
Text(detail).font(.callout).foregroundStyle(.secondary)
}
Spacer()
}
.padding()
.background(Color.gray.opacity(0.05))
.cornerRadius(8)
.contentShape(Rectangle())
}
.buttonStyle(.plain)
}
private var allChecked: Bool {
fmiConfirmed && sdpConfirmed && appleIdConfirmed && rebreakAppInstalled
}
private var navigationBar: some View {
HStack {
Button("Zurück") { model.goTo(.welcome) }
.buttonStyle(.bordered)
Spacer()
Button("Supervisieren starten →") { model.advance() }
.buttonStyle(.borderedProminent)
.disabled(!allChecked)
}
}
}