現在、テキスト読み上げを必要とする Java アプリケーション (チャット アプリ) を作成しています。少しオンラインで調べたところ、FreeTTS が適切なオプションであることがわかりました。なんとか動作させることができました。現在、メイン UI スレッドをスレッドで実行しているため、メイン UI スレッドがブロックされるのを防いでいます。
唯一の問題は、FreeTTS では一度に 1 つの音声しか話せないことです (複数のインスタンスとして複数のスレッドにあるにもかかわらず)。これを回避して、複数のユーザーが一度に話せるようにする方法はありますか? これは私がこれまで持っている私のコードです。
import lombok.extern.slf4j.Slf4j;
import java.beans.PropertyVetoException;
import java.util.Locale;
import java.util.Random;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.speech.AudioException;
import javax.speech.Central;
import javax.speech.EngineException;
import javax.speech.EngineStateError;
import javax.speech.synthesis.Synthesizer;
import javax.speech.synthesis.SynthesizerModeDesc;
import javax.speech.synthesis.Voice;
@Slf4j
public enum Speech implements SpeechAgent {
Instance;
private static Random pool = new Random();
private static Synthesizer createSynthesizer() {
try {
System.setProperty("freetts.voices", "com.sun.speech.freetts.en.us.cmu_us_kal.KevinVoiceDirectory");
SynthesizerModeDesc desc = new SynthesizerModeDesc(Locale.US);
Central.registerEngineCentral("com.sun.speech.freetts.jsapi.FreeTTSEngineCentral");
Synthesizer synthesizer = Central.createSynthesizer(desc);
synthesizer.allocate();
synthesizer.resume();
Voice[] voices = null;
while (voices == null) {
voices = ((SynthesizerModeDesc) synthesizer.getEngineModeDesc()).getVoices();
}
synthesizer.getSynthesizerProperties().setVoice(voices[pool.nextInt(voices.length)]);
return synthesizer;
} catch (Exception e) {
log.error(String.format("Failed to load synthesizer: %s", e.toString()));
}
return null;
}
public void speak(String input) {
new Thread(new Runnable() {
public void run() {
Synthesizer synthesizer = createSynthesizer();
if (synthesizer == null) {
return;
}
synthesizer.speakPlainText(input.replaceAll("[^a-zA-Z ]+", ""), null);
}
}).start();
}
}
スレッドでスローされたエラー:
Trouble while processing utterance java.lang.IllegalStateException: Syllable relation has already been set
java.lang.IllegalStateException: Syllable relation has already been set
at com.sun.speech.freetts.Segmenter.processUtterance(Segmenter.java:56)
at com.sun.speech.freetts.Voice.runProcessor(Voice.java:595)
at com.sun.speech.freetts.Voice.processUtterance(Voice.java:414)
at com.sun.speech.freetts.Voice.speak(Voice.java:289)
at com.sun.speech.freetts.jsapi.FreeTTSSynthesizer$OutputHandler.outputItem(FreeTTSSynthesizer.java:695)
at com.sun.speech.freetts.jsapi.FreeTTSSynthesizer$OutputHandler.run(FreeTTSSynthesizer.java:622)