feat(onboarding,diga): TTS auto-play preference + 90 more DiGA test codes

## TTS Auto-Play Preference

User-Request: wenn Voice einmal aktiviert, soll Lyra auf jeder Slide
automatisch sprechen — nicht jede Slide extra antippen.

- stores/lyraVoice.ts: zustand-store mit AsyncStorage-Persistence
  (@rebreak/lyraVoiceEnabled). Default OFF.
- LyraBubble auto-plays on text-change wenn enabled
- Audio-Button toggled die Preference + stoppt current playback
- Visuell: Button ist orange-filled wenn voice ON, ghost-bordered wenn OFF
- Icon: volume-mute-outline (OFF) / volume-medium / hourglass / stop
- Cleanup beim Unmount (stopLyraSpeech) + bei text-change

Initialisiert via init() in app/_layout.tsx (analog language/theme/appLock).

Locale-keys: audio_play → "Stimme einschalten", neu audio_disable → "Stimme
ausschalten" in 4 Sprachen.

## DiGA Test Codes 011-100

Aktuell 10 Codes (REBREAK-TEST-001..010), aber 100 Android-Tester kommen
morgen onboarding. Migration 20260518_extend_diga_test_codes seeded 90
zusätzliche Codes via generate_series(11, 100) + LPAD-Padding.

- Label: 'test_batch_2026-05-android' für Auditbarkeit (vs '...2026-05'
  für die ersten 10)
- grants_plan: 'legend' wie die ersten 10
- ON CONFLICT DO NOTHING — idempotent

Distribution-Pattern: Tester N kriegt Code REBREAK-TEST-<NNN-padded>.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
chahinebrini 2026-05-17 22:39:18 +02:00
parent 56bb59915d
commit ac605dce33
8 changed files with 134 additions and 39 deletions

View File

@ -25,6 +25,7 @@ import { useRealtimeDebugStore } from '../stores/realtimeDebug';
import { useColors } from '../lib/theme'; import { useColors } from '../lib/theme';
import { useLanguageStore } from '../stores/language'; import { useLanguageStore } from '../stores/language';
import { useAppLockStore } from '../stores/appLock'; import { useAppLockStore } from '../stores/appLock';
import { useLyraVoiceStore } from '../stores/lyraVoice';
import { BrandSplash } from '../components/BrandSplash'; import { BrandSplash } from '../components/BrandSplash';
import { AppLockGate } from '../components/AppLockGate'; import { AppLockGate } from '../components/AppLockGate';
import { DeviceLimitReachedSheet } from '../components/DeviceLimitReachedSheet'; import { DeviceLimitReachedSheet } from '../components/DeviceLimitReachedSheet';
@ -57,6 +58,7 @@ function RootLayoutInner() {
const colorScheme = useThemeStore((s) => s.colorScheme); const colorScheme = useThemeStore((s) => s.colorScheme);
const initLanguage = useLanguageStore((s) => s.init); const initLanguage = useLanguageStore((s) => s.init);
const initAppLock = useAppLockStore((s) => s.init); const initAppLock = useAppLockStore((s) => s.init);
const initLyraVoice = useLyraVoiceStore((s) => s.init);
const appLockReady = useAppLockStore((s) => s.ready); const appLockReady = useAppLockStore((s) => s.ready);
const initRealtimeDebug = useRealtimeDebugStore((s) => s.init); const initRealtimeDebug = useRealtimeDebugStore((s) => s.init);
const colors = useColors(); const colors = useColors();
@ -72,6 +74,7 @@ function RootLayoutInner() {
initTheme(); initTheme();
initLanguage(); initLanguage();
initAppLock(); initAppLock();
initLyraVoice();
if (__DEV__) initRealtimeDebug(); if (__DEV__) initRealtimeDebug();
}, []); }, []);

View File

@ -10,15 +10,20 @@ import {
stopLyraSpeech, stopLyraSpeech,
type LyraSpeechStatus, type LyraSpeechStatus,
} from '../../lib/lyraSpeech'; } from '../../lib/lyraSpeech';
import { useLyraVoiceStore } from '../../stores/lyraVoice';
/** /**
* Lyra-Mascot (animiertes Rive-Avatar) links + Speech-Bubble rechts. * Lyra-Mascot (animiertes Rive-Avatar) + Speech-Bubble + TTS-Audio-Button.
* *
* Plus: kleiner Audio-Button rechts oben in der Bubble User kann den Lyra- * Auto-Play-Pattern (User-Preference, persistent via useLyraVoiceStore):
* Text via TTS hören lassen. DiGA-Accessibility-relevant (Screen-Reader- * - Voice OFF (default): kein Auto-Play, Audio-Button-Tap aktiviert Voice
* Alternative + Lese-Hürden-Mitigation). * + spielt aktuellen Text.
* - Voice ON: Bubble spielt sich auf jeder Slide automatisch ab beim
* Text-Change. Audio-Button-Tap stoppt aktuelles Playback + disabled
* die Preference (= kein Auto-Play mehr).
* *
* Fade+slide-in beim Mount und bei text-change (key-prop verwenden für Re-Animate). * Cleanup beim Unmount + bei text-change (Slide-Switch vorheriger Sound
* wird abgebrochen).
*/ */
export function LyraBubble({ export function LyraBubble({
text, text,
@ -32,7 +37,11 @@ export function LyraBubble({
const opacity = useRef(new Animated.Value(0)).current; const opacity = useRef(new Animated.Value(0)).current;
const translateX = useRef(new Animated.Value(-12)).current; const translateX = useRef(new Animated.Value(-12)).current;
const [speech, setSpeech] = useState<LyraSpeechStatus>('idle'); const [speech, setSpeech] = useState<LyraSpeechStatus>('idle');
const voiceEnabled = useLyraVoiceStore((s) => s.enabled);
const voiceReady = useLyraVoiceStore((s) => s.ready);
const toggleVoice = useLyraVoiceStore((s) => s.toggle);
// Fade-in beim Mount + text-change
useEffect(() => { useEffect(() => {
opacity.setValue(0); opacity.setValue(0);
translateX.setValue(-12); translateX.setValue(-12);
@ -52,34 +61,50 @@ export function LyraBubble({
]).start(); ]).start();
}, [text, opacity, translateX]); }, [text, opacity, translateX]);
// Cleanup beim Unmount — sonst spielt der Sound weiter wenn User Slide wechselt // Auto-Play wenn voice enabled + Bubble-Text wechselt (= neue Slide).
// Stoppt vorheriges Playback automatisch via stopLyraSpeech() in playLyraSpeech.
useEffect(() => {
if (!voiceReady) return; // warte bis AsyncStorage geladen
if (!voiceEnabled) {
// Falls voice grad disabled wurde während etwas spielte: stoppen.
setSpeech('idle');
void stopLyraSpeech();
return;
}
void playLyraSpeech(text, i18n.language || 'de', setSpeech);
}, [text, voiceEnabled, voiceReady]);
// Cleanup beim Unmount — sonst spielt der Sound weiter wenn /onboarding entladen wird
useEffect(() => { useEffect(() => {
return () => { return () => {
void stopLyraSpeech(); void stopLyraSpeech();
}; };
}, []); }, []);
// Stoppt auch wenn der Bubble-Text wechselt (= neuer Slide angezeigt)
useEffect(() => {
setSpeech('idle');
void stopLyraSpeech();
}, [text]);
async function togglePlayback() { async function togglePlayback() {
if (speech === 'playing' || speech === 'loading') { // Wenn voice OFF: enable (auto-trigger Auto-Play-Effect oben)
await stopLyraSpeech(); // Wenn voice ON: disable + stop current
if (voiceEnabled) {
setSpeech('idle'); setSpeech('idle');
return; await stopLyraSpeech();
} }
await playLyraSpeech(text, i18n.language || 'de', setSpeech); await toggleVoice();
// Wenn jetzt enabled: useEffect oben startet das playback automatisch
} }
const a11yLabel = const a11yLabel = voiceEnabled
speech === 'playing' ? speech === 'playing'
? t('onboarding.lyra.audio_stop') ? t('onboarding.lyra.audio_stop')
: t('onboarding.lyra.audio_disable')
: t('onboarding.lyra.audio_play');
const iconName: keyof typeof Ionicons.glyphMap = !voiceEnabled
? 'volume-mute-outline'
: speech === 'playing'
? 'stop'
: speech === 'loading' : speech === 'loading'
? t('onboarding.lyra.audio_loading') ? 'hourglass'
: t('onboarding.lyra.audio_play'); : 'volume-medium';
return ( return (
<View style={{ flexDirection: 'row', alignItems: 'flex-start', gap: 12 }}> <View style={{ flexDirection: 'row', alignItems: 'flex-start', gap: 12 }}>
@ -100,7 +125,7 @@ export function LyraBubble({
borderRadius: 16, borderRadius: 16,
paddingVertical: 14, paddingVertical: 14,
paddingLeft: 16, paddingLeft: 16,
paddingRight: 50, // Platz für den Audio-Button rechts paddingRight: 50, // Platz für Audio-Button rechts
position: 'relative', position: 'relative',
}} }}
> >
@ -115,7 +140,7 @@ export function LyraBubble({
{text} {text}
</Text> </Text>
{/* Audio-Button rechts oben — kompakt, dezent */} {/* Audio-Toggle rechts oben — bestimmt globale Voice-Preference */}
<TouchableOpacity <TouchableOpacity
onPress={togglePlayback} onPress={togglePlayback}
hitSlop={8} hitSlop={8}
@ -129,21 +154,17 @@ export function LyraBubble({
width: 34, width: 34,
height: 34, height: 34,
borderRadius: 17, borderRadius: 17,
backgroundColor: colors.brandOrange, backgroundColor: voiceEnabled ? colors.brandOrange : colors.surface,
borderWidth: voiceEnabled ? 0 : 1,
borderColor: colors.border,
alignItems: 'center', alignItems: 'center',
justifyContent: 'center', justifyContent: 'center',
}} }}
> >
<Ionicons <Ionicons
name={ name={iconName}
speech === 'playing'
? 'stop'
: speech === 'loading'
? 'hourglass'
: 'volume-medium'
}
size={16} size={16}
color="#ffffff" color={voiceEnabled ? '#ffffff' : colors.textMuted}
/> />
</TouchableOpacity> </TouchableOpacity>
</View> </View>

View File

@ -371,9 +371,10 @@
"protection_url": { "body": "ستظهر نافذة iOS. اضغط «السماح» — الزر السفلي (وليس الأزرق الكبير في الأعلى — هذا هو الفخ)." }, "protection_url": { "body": "ستظهر نافذة iOS. اضغط «السماح» — الزر السفلي (وليس الأزرق الكبير في الأعلى — هذا هو الفخ)." },
"protection_lock": { "body": "الآن قفل التطبيق. iOS يطلب الوصول إلى مدة استخدام الجهاز — اضغط «متابعة»، مرة أخرى الزر السفلي (وليس الأزرق)." }, "protection_lock": { "body": "الآن قفل التطبيق. iOS يطلب الوصول إلى مدة استخدام الجهاز — اضغط «متابعة»، مرة أخرى الزر السفلي (وليس الأزرق)." },
"done": { "body": "تم. اليوم الأول من سلسلتك — ولست وحدك." }, "done": { "body": "تم. اليوم الأول من سلسلتك — ولست وحدك." },
"audio_play": "اقرأ النص بصوت عالٍ", "audio_play": "تفعيل الصوت",
"audio_loading": "جاري تحميل الصوت...", "audio_loading": "جاري تحميل الصوت...",
"audio_stop": "إيقاف التشغيل" "audio_stop": "إيقاف التشغيل",
"audio_disable": "إيقاف الصوت"
}, },
"welcome": { "welcome": {
"cta_primary": "هيا نبدأ", "cta_primary": "هيا نبدأ",

View File

@ -371,9 +371,10 @@
"protection_url": { "body": "Gleich kommt ein iOS-Dialog. Tippe \"Erlauben\" — den unteren Button (nicht den großen blauen oben — das ist die Falle)." }, "protection_url": { "body": "Gleich kommt ein iOS-Dialog. Tippe \"Erlauben\" — den unteren Button (nicht den großen blauen oben — das ist die Falle)." },
"protection_lock": { "body": "Jetzt der App-Schutz. iOS fragt nach Bildschirmzeit-Zugriff — tippe \"Fortfahren\", wieder den unteren Button (nicht den blauen)." }, "protection_lock": { "body": "Jetzt der App-Schutz. iOS fragt nach Bildschirmzeit-Zugriff — tippe \"Fortfahren\", wieder den unteren Button (nicht den blauen)." },
"done": { "body": "Geschafft. Tag 1 deiner neuen Streak — und du gehst nicht allein." }, "done": { "body": "Geschafft. Tag 1 deiner neuen Streak — und du gehst nicht allein." },
"audio_play": "Text vorlesen", "audio_play": "Stimme einschalten",
"audio_loading": "Lade Stimme...", "audio_loading": "Lade Stimme...",
"audio_stop": "Wiedergabe stoppen" "audio_stop": "Wiedergabe stoppen",
"audio_disable": "Stimme ausschalten"
}, },
"welcome": { "welcome": {
"cta_primary": "Los geht's", "cta_primary": "Los geht's",

View File

@ -371,9 +371,10 @@
"protection_url": { "body": "An iOS dialog is coming. Tap \"Allow\" — the bottom button (not the big blue one on top — that's the trap)." }, "protection_url": { "body": "An iOS dialog is coming. Tap \"Allow\" — the bottom button (not the big blue one on top — that's the trap)." },
"protection_lock": { "body": "Now the app lock. iOS asks for Screen Time access — tap \"Continue\", again the bottom button (not the blue one)." }, "protection_lock": { "body": "Now the app lock. iOS asks for Screen Time access — tap \"Continue\", again the bottom button (not the blue one)." },
"done": { "body": "Done. Day 1 of your new streak — and you're not walking alone." }, "done": { "body": "Done. Day 1 of your new streak — and you're not walking alone." },
"audio_play": "Read text aloud", "audio_play": "Enable voice",
"audio_loading": "Loading voice...", "audio_loading": "Loading voice...",
"audio_stop": "Stop playback" "audio_stop": "Stop playback",
"audio_disable": "Disable voice"
}, },
"welcome": { "welcome": {
"cta_primary": "Let's go", "cta_primary": "Let's go",

View File

@ -369,9 +369,10 @@
"protection_url": { "body": "Une fenêtre iOS va apparaître. Touche « Autoriser » — le bouton du bas (pas le grand bleu en haut — c'est le piège)." }, "protection_url": { "body": "Une fenêtre iOS va apparaître. Touche « Autoriser » — le bouton du bas (pas le grand bleu en haut — c'est le piège)." },
"protection_lock": { "body": "Maintenant le verrou d'app. iOS demande l'accès à Temps d'écran — touche « Continuer », encore le bouton du bas (pas le bleu)." }, "protection_lock": { "body": "Maintenant le verrou d'app. iOS demande l'accès à Temps d'écran — touche « Continuer », encore le bouton du bas (pas le bleu)." },
"done": { "body": "Voilà. Jour 1 de ta nouvelle série — et tu n'es pas seul." }, "done": { "body": "Voilà. Jour 1 de ta nouvelle série — et tu n'es pas seul." },
"audio_play": "Lire le texte", "audio_play": "Activer la voix",
"audio_loading": "Chargement de la voix...", "audio_loading": "Chargement de la voix...",
"audio_stop": "Arrêter la lecture" "audio_stop": "Arrêter la lecture",
"audio_disable": "Désactiver la voix"
}, },
"welcome": { "welcome": {
"cta_primary": "On y va", "cta_primary": "On y va",

View File

@ -0,0 +1,51 @@
import { create } from 'zustand';
import AsyncStorage from '@react-native-async-storage/async-storage';
/**
* Persistente User-Preference für Lyra-TTS-Auto-Play.
*
* Default OFF User aktiviert einmal in der Welcome-Slide via Audio-Button,
* danach playt Lyra auf jeder Slide automatisch. Erneutes Tappen disabled
* + stoppt aktuelles Playback.
*
* Persistence: AsyncStorage @rebreak/lyraVoiceEnabled. Init beim App-Start
* via `init()` (analog zu language/theme/appLock-Stores).
*/
const STORAGE_KEY = '@rebreak/lyraVoiceEnabled';
type LyraVoiceState = {
enabled: boolean;
ready: boolean;
init: () => Promise<void>;
setEnabled: (enabled: boolean) => Promise<void>;
toggle: () => Promise<void>;
};
export const useLyraVoiceStore = create<LyraVoiceState>((set, get) => ({
enabled: false,
ready: false,
init: async () => {
try {
const stored = await AsyncStorage.getItem(STORAGE_KEY);
set({ enabled: stored === '1', ready: true });
} catch {
set({ enabled: false, ready: true });
}
},
setEnabled: async (enabled: boolean) => {
set({ enabled });
try {
await AsyncStorage.setItem(STORAGE_KEY, enabled ? '1' : '0');
} catch {
// ignore — in-memory state ist wichtiger
}
},
toggle: async () => {
const next = !get().enabled;
await get().setEnabled(next);
},
}));

View File

@ -0,0 +1,16 @@
-- Extend DiGA-Test-Codes from 10 → 100 für 100 Android-Tester-Onboarding.
--
-- REBREAK-TEST-011 bis REBREAK-TEST-100 — gleicher Pattern wie 001-010
-- aus Migration 20260517_add_diga_codes.
--
-- ON CONFLICT DO NOTHING — idempotent, re-Run safe. Bestehende 001-010
-- bleiben unverändert.
INSERT INTO "rebreak"."diga_codes" ("code", "label", "grants_plan", "notes")
SELECT
'REBREAK-TEST-' || LPAD(n::text, 3, '0'),
'test_batch_2026-05-android',
'legend',
'Android-Tester-Cohort 2026-05 (single-use, reset via SQL)'
FROM generate_series(11, 100) AS n
ON CONFLICT ("code") DO NOTHING;