diff --git a/android/src/screens/ChatScreen.tsx b/android/src/screens/ChatScreen.tsx index fb26333..f6c9200 100644 --- a/android/src/screens/ChatScreen.tsx +++ b/android/src/screens/ChatScreen.tsx @@ -535,6 +535,7 @@ const ChatScreen: React.FC = () => { audioRequestId, ...(location && { location }), }); + scheduleStaleAudioCleanup(audioRequestId); // resume() wird durch onPlaybackFinished nach ARIAs Antwort getriggert. } else { // Kein Speech im Window → Konversation beenden (Ohr geht aus oder @@ -656,6 +657,25 @@ const ChatScreen: React.FC = () => { // --- Nachricht senden --- + // Aufraeumen von "verarbeitet"-Placeholder die nie ein STT-Result bekommen + // haben (leere Aufnahme, Wake-Word-Echo, STT-Fehler etc). Nach 30s werden + // sie automatisch entfernt damit nicht-erkannte Aufnahmen nicht den State + // verstopfen + naechste echte Aufnahmen die richtige Bubble ersetzen koennen. + const scheduleStaleAudioCleanup = useCallback((audioRequestId: string) => { + setTimeout(() => { + setMessages(prev => { + const idx = prev.findIndex(m => + m.audioRequestId === audioRequestId && + m.text.includes('Spracheingabe wird verarbeitet') + ); + if (idx < 0) return prev; + console.log('[Chat] Sprachnachricht ohne STT-Result entfernt: %s', audioRequestId); + ToastAndroid.show('Sprachnachricht nicht erkannt — entfernt', ToastAndroid.SHORT); + return prev.filter((_, i) => i !== idx); + }); + }, 30000); + }, []); + const sendTextMessage = useCallback(async () => { const text = inputText.trim(); @@ -743,7 +763,8 @@ const ChatScreen: React.FC = () => { audioRequestId, ...(location && { location }), }); - }, [getCurrentLocation, interruptAriaIfBusy]); + scheduleStaleAudioCleanup(audioRequestId); + }, [getCurrentLocation, interruptAriaIfBusy, scheduleStaleAudioCleanup]); // Datei auswaehlen → zur Pending-Liste hinzufuegen const handleFileSelected = useCallback(async (file: FileData) => {