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
84 lines
2.9 KiB
Swift
84 lines
2.9 KiB
Swift
import SwiftUI
|
|
|
|
struct DoneView: View {
|
|
@Environment(WizardModel.self) private var model
|
|
|
|
var body: some View {
|
|
VStack(spacing: 24) {
|
|
Image(systemName: "checkmark.seal.fill")
|
|
.font(.system(size: 80))
|
|
.foregroundStyle(.green)
|
|
|
|
Text("Schutz aktiv")
|
|
.font(.largeTitle).bold()
|
|
|
|
Text("Dein iPhone ist jetzt an ReBreak gebunden. Casino-Domains werden via NEFilter blockiert — auch wenn du es willst.")
|
|
.multilineTextAlignment(.center)
|
|
.foregroundStyle(.secondary)
|
|
.padding(.horizontal, 40)
|
|
|
|
statusSummary
|
|
|
|
cooldownNote
|
|
|
|
VStack(spacing: 8) {
|
|
Button("ReBreak öffnen") {
|
|
if let url = URL(string: "rebreak://") {
|
|
NSWorkspace.shared.open(url)
|
|
}
|
|
}
|
|
.buttonStyle(.borderedProminent)
|
|
.controlSize(.large)
|
|
|
|
Button("Wizard schließen / Neuer Bind") {
|
|
model.reset()
|
|
}
|
|
.buttonStyle(.plain)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
.padding(40)
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
}
|
|
|
|
private var statusSummary: some View {
|
|
VStack(alignment: .leading, spacing: 8) {
|
|
statusRow(label: "Supervised", on: model.device?.isSupervised == true)
|
|
statusRow(label: "MDM-Enrolled", on: model.device?.isEnrolled == true)
|
|
statusRow(label: "App managed (nicht löschbar)", on: model.device?.isManaged == true)
|
|
statusRow(label: "NEFilter aktiv (Casino-Block)", on: model.device?.isFilterActive == true)
|
|
}
|
|
.padding()
|
|
.frame(maxWidth: 400)
|
|
.background(Color.green.opacity(0.05))
|
|
.cornerRadius(8)
|
|
}
|
|
|
|
private func statusRow(label: String, on: Bool) -> some View {
|
|
HStack {
|
|
Image(systemName: on ? "checkmark.circle.fill" : "circle")
|
|
.foregroundStyle(on ? .green : .gray)
|
|
Text(label)
|
|
Spacer()
|
|
}
|
|
.font(.callout)
|
|
}
|
|
|
|
private var cooldownNote: some View {
|
|
HStack(alignment: .top, spacing: 8) {
|
|
Image(systemName: "clock.badge.exclamationmark")
|
|
.foregroundStyle(.orange)
|
|
VStack(alignment: .leading, spacing: 4) {
|
|
Text("7-Tage-Cooldown").bold()
|
|
Text("Wenn du den Schutz aufheben willst, gibt's eine 7-Tage-Wartezeit. Das ist Absicht — die Bindung soll deinen impulsiven 'jetzt-doch-zocken'-Moment überdauern.")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
.padding(10)
|
|
.frame(maxWidth: 400, alignment: .leading)
|
|
.background(Color.orange.opacity(0.08))
|
|
.cornerRadius(6)
|
|
}
|
|
}
|