私のAndroidアプリケーションには話すテキスト機能があり、複数のアクティビティがこの機能を使用しています。そこで、これを簡単にするために静的ヘルパークラスを作成しました。
import Java.util.Locale;
import Android.content.Context;
import Android.speech.tts.TextToSpeech;
import Android.speech.tts.TextToSpeech.OnInitListener;
import Android.util.Log;
public class TextToSpeechController implements OnInitListener {
private static final String TAG = "TextToSpeechController";
private TextToSpeech myTTS;
private String textToSpeak;
private Context context;
private static TextToSpeechController singleton;
public static TextToSpeechController getInstance(Context ctx) {
if (singleton == null)
singleton = new TextToSpeechController(ctx);
return singleton;
}
private TextToSpeechController(Context ctx) {
context = ctx;
}
public void speak(String text) {
textToSpeak = text;
if (myTTS == null) {
// currently can't change Locale until speech ends
try {
// Initialize text-to-speech. This is an asynchronous operation.
// The OnInitListener (second argument) is called after
// initialization completes.
myTTS = new TextToSpeech(context, this);
} catch (Exception e) {
e.printStackTrace();
}
}
sayText();
}
public void onInit(int initStatus) {
if (initStatus == TextToSpeech.SUCCESS) {
if (myTTS.isLanguageAvailable(Locale.UK) == TextToSpeech.LANG_AVAILABLE)
myTTS.setLanguage(Locale.UK);
}
// status can be either TextToSpeech.SUCCESS or TextToSpeech.ERROR.
if (initStatus == TextToSpeech.SUCCESS) {
int result = myTTS.setLanguage(Locale.UK);
if (result == TextToSpeech.LANG_MISSING_DATA
|| result == TextToSpeech.LANG_NOT_SUPPORTED) {
Log.e(TAG, "TTS missing or not supported (" + result + ")");
// Language data is missing or the language is not supported.
// showError(R.string.tts_lang_not_available);
} else {
// Initialization failed.
Log.e(TAG, "Error occured");
}
}
}
private void sayText() {
// ask TTs to say the text
myTTS.speak(this.textToSpeak, TextToSpeech.QUEUE_FLUSH, null);
}
public void stopTTS() {
if (myTTS != null) {
myTTS.shutdown();
myTTS.stop();
myTTS = null;
}
}
}
私はこのようなヘルパークラスを使用しています。
TextToSpeechController.getInstance(this).speak(readableMessage);
しかし、LogCatで次のようなエラーが表示されることがあります。
"speak:failed not bound to tts engine"
例外はスローされませんが、TTSによって読み取られるものはありません。 BroadcastReceiverからアクティビティをトリガーすると、このエラーが発生することに気付きました。それ以外の場合、アプリケーションコードから手動でアクティビティを開くと、問題なく機能します。
これがBroadcastReceiverコードです
private final BroadcastReceiver mHandleMessageReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String newMessage = intent.getExtras().getString(EXTRA_MESSAGE);
String readableMessage = intent.getExtras().getString(READABLE_MESSAGE);
Bundle b = new Bundle();
b.putString(EXTRA_MESSAGE, newMessage);
b.putString(READABLE_MESSAGE, readableMessage);
Intent newIntent = new Intent("Android.intent.action.MAIN");
newIntent.setClass(context, Speak.class);
newIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
newIntent.putExtra("MessageReceived", b);
newIntent.putExtra("CallType", CallType.NOTIFICATION);
context.startActivity(newIntent);
}
};
コードは、TextToSpeechクラスのコンストラクターからの非同期の戻りを待っていません。非同期だというコメントもあります。なぜReceiverで常に失敗するのに、アクティビティでは失敗するのかはわかりません。それ以外の場合は、フォアグラウンドタスクとしてのアクティビティの優先度が高く、sayTextを呼び出す前に完了します。
新しいTTSオブジェクトを新規作成する必要がある場合は、init呼び出しが戻るまでsayTextを呼び出すのを待つ必要があります。
これが私がテキスト読み上げに使用するコードです。私のコードでは、speakWords( "saysomething");と入力するだけです。どんな活動でも。
public class VoiceRecognition extends Activity implements OnClickListener,
OnInitListener {
public static final int VOICE_RECOGNITION_REQUEST_CODE = 1234;
public Button speakButton;
// TTS object
public TextToSpeech myTTS;
// status check code
public int MY_DATA_CHECK_CODE = 0;
// setup TTS
public void onInit(int initStatus) {
// check for successful instantiation
if (initStatus == TextToSpeech.SUCCESS) {
if (myTTS.isLanguageAvailable(Locale.US) == TextToSpeech.LANG_AVAILABLE)
myTTS.setLanguage(Locale.US);
} else if (initStatus == TextToSpeech.ERROR) {
Toast.makeText(this, "Sorry! Text To Speech failed...",
Toast.LENGTH_LONG).show();
}
}
/**
* Called with the activity is first created.
*/
@Override
public void onCreate(Bundle voiceinput) {
super.onCreate(voiceinput);
// Inflate our UI from its XML layout description.
setContentView(R.layout.voice_recognition);
// check for TTS data
Intent checkTTSIntent = new Intent();
checkTTSIntent.setAction(TextToSpeech.Engine.ACTION_CHECK_TTS_DATA);
startActivityForResult(checkTTSIntent, MY_DATA_CHECK_CODE);
}
// speak the user text
public void speakWords(String speech) {
// speak straight away
myTTS.speak(speech, TextToSpeech.QUEUE_FLUSH, null);
}
public void onClick(View v) {
speakWords("hello";
}
/**
* Handle the results from the recognition activity.
*/
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == MY_DATA_CHECK_CODE) {
if (resultCode == TextToSpeech.Engine.CHECK_VOICE_DATA_PASS) {
// the user has the necessary data - create the TTS
myTTS = new TextToSpeech(this, this);
} else {
// no data - install it now
Intent installTTSIntent = new Intent();
installTTSIntent
.setAction(TextToSpeech.Engine.ACTION_INSTALL_TTS_DATA);
startActivity(installTTSIntent);
}
}
super.onActivityResult(requestCode, resultCode, data);
}
@Override
protected void onDestroy() {
super.onDestroy();
myTTS.shutdown();
}