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:
parent
56bb59915d
commit
ac605dce33
@ -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();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|||||||
@ -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>
|
||||||
|
|||||||
@ -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": "هيا نبدأ",
|
||||||
|
|||||||
@ -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",
|
||||||
|
|||||||
@ -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",
|
||||||
|
|||||||
@ -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",
|
||||||
|
|||||||
51
apps/rebreak-native/stores/lyraVoice.ts
Normal file
51
apps/rebreak-native/stores/lyraVoice.ts
Normal 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);
|
||||||
|
},
|
||||||
|
}));
|
||||||
@ -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;
|
||||||
Loading…
x
Reference in New Issue
Block a user