js_update_ipa_output = """ /** * Updates the CSS text of the given text based on the correctness of each letter. * * @param text - The text to be displayed. * @param letters - A string representing the correctness of each letter in the text. * @param idxSelectedWord - The index of the selected word to be underlined. */ function updateCssText(text, letters, idxSelectedWord) { let wordsArr = text.split(" ") let lettersWordsArr = letters.split(" ") let speechOutputContainer = document.querySelector('#speech-output'); speechOutputContainer.textContent = "" for (let idx in wordsArr) { let word = wordsArr[idx] let letterIsCorrect = lettersWordsArr[idx] for (let idx1 in word) { let letterCorrect = letterIsCorrect[idx1] == "1" let containerLetter = document.createElement("span") let color = letterCorrect ? 'green' : "red" containerLetter.style.cssText = idx == idxSelectedWord ? `color: ${color}; text-decoration-line: underline;` : `color: ${color};` containerLetter.innerText = word[idx1]; speechOutputContainer.appendChild(containerLetter) } let containerSpace = document.createElement("span") containerSpace.textContent = " " speechOutputContainer.appendChild(containerSpace) } } """ js_play_audio = """ /** * Plays the given text as audio using the Web Speech API. * * @param text - The text to be spoken. * @param language - The language code for the speech synthesis (e.g., 'en' for English, 'de' for German). * @param sleepTime - Optional. The time in seconds to wait before starting the speech synthesis. Default is 0. */ function playAudio(text, language, sleepTime = 0) { let voice_idx = 0; let voice_synth = null; let synth = window.speechSynthesis; let voice_lang; let sleepTimeAdditional = 500; // for some reason using this with a default input argument give an object instead of the correct number function sleep (time) { return new Promise((resolve) => setTimeout(resolve, time)); } function setSpeech() { return new Promise( function (resolve, reject) { let id; id = setInterval(() => { if (synth.getVoices().length !== 0) { resolve(synth.getVoices()); clearInterval(id); } }, 10); } ) } switch (language) { case 'de': voice_lang = 'de-DE'; break; case 'en': voice_lang = 'en-US'; break; default: const msg = `Error: language ${language} not valid!` console.error(msg); alert(msg) throw new Error(msg) } let s = setSpeech(); s.then((voices) => { let voicesSynth = voices.filter(voice => voice.lang === voice_lang); if (voicesSynth.length === 0) { console.error(`No voice found for language ${voice_lang}, retry for less restrictive check (startsWith)...`) voicesSynth = voices.filter(voice => voice.lang.startsWith(language)); } if (voicesSynth.length === 0) { const msg = `Error: no voice found for language ${voice_lang} / ${language}, you should use the Text-To-Speech backend feature...` console.error(msg); alert(msg) throw new Error(msg) } var utterThis = new SpeechSynthesisUtterance(text); utterThis.voice = voicesSynth[0]; utterThis.rate = 0.7; if (sleepTime > 0) { sleepTime *= 1000; sleepTime += sleepTimeAdditional; } console.log("start js_play_audio:: sleepTime:", sleepTime, "#") sleep(sleepTime).then(() => { synth.speak(utterThis); }) // todo: capture audio from speech synthesis to reuse on the frontend // https://stackoverflow.com/questions/45003548/how-to-capture-generated-audio-from-window-speechsynthesis-speak-call }); } """ head_driver_tour = """ """